directory-handle: make lttng_directory_handle_stat public
[lttng-tools.git] / src / common / runas.c
CommitLineData
60b6c79c
MD
1/*
2 * Copyright (C) 2011 - David Goulet <david.goulet@polymtl.ca>
3 * Mathieu Desnoyers <mathieu.desnoyers@efficios.com>
93bed9fe 4 * 2019 - Jérémie Galarneau <jeremie.galarneau@efficios.com>
60b6c79c 5 *
d14d33bf
AM
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License, version 2 only,
8 * as published by the Free Software Foundation.
60b6c79c
MD
9 *
10 * This program is distributed in the hope that it will be useful, but WITHOUT
11 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
d14d33bf 12 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
60b6c79c
MD
13 * more details.
14 *
d14d33bf
AM
15 * You should have received a copy of the GNU General Public License along
16 * with this program; if not, write to the Free Software Foundation, Inc.,
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
60b6c79c
MD
18 */
19
6c1c0768 20#define _LGPL_SOURCE
60b6c79c
MD
21#include <errno.h>
22#include <limits.h>
23#include <stdio.h>
24#include <stdlib.h>
25#include <string.h>
26#include <sys/wait.h>
27#include <sys/types.h>
28#include <sys/stat.h>
29#include <unistd.h>
30#include <fcntl.h>
c2b75c49 31#include <sched.h>
0452bf08 32#include <signal.h>
749b7a0c 33#include <assert.h>
e1055edb 34#include <signal.h>
60b6c79c 35
0ef03255 36#include <common/lttng-kernel.h>
90e535ef 37#include <common/common.h>
3fd15a74 38#include <common/utils.h>
e8fa9fb0 39#include <common/compat/getenv.h>
e1055edb 40#include <common/compat/prctl.h>
2038dd6c
JG
41#include <common/unix.h>
42#include <common/defaults.h>
241e0a5a
FD
43#include <common/lttng-elf.h>
44
45#include <lttng/constant.h>
60b6c79c 46
0857097f
DG
47#include "runas.h"
48
7567352f 49struct run_as_data;
fe9f7760
FD
50struct run_as_ret;
51typedef int (*run_as_fct)(struct run_as_data *data, struct run_as_ret *ret_value);
c2b75c49 52
93bed9fe
JG
53enum run_as_cmd {
54 RUN_AS_MKDIR,
55 RUN_AS_MKDIRAT,
56 RUN_AS_MKDIR_RECURSIVE,
57 RUN_AS_MKDIRAT_RECURSIVE,
58 RUN_AS_OPEN,
59 RUN_AS_OPENAT,
60 RUN_AS_UNLINK,
61 RUN_AS_UNLINKAT,
62 RUN_AS_RMDIR,
63 RUN_AS_RMDIRAT,
64 RUN_AS_RMDIR_RECURSIVE,
65 RUN_AS_RMDIRAT_RECURSIVE,
66 RUN_AS_RENAME,
67 RUN_AS_RENAMEAT,
68 RUN_AS_EXTRACT_ELF_SYMBOL_OFFSET,
69 RUN_AS_EXTRACT_SDT_PROBE_OFFSETS,
60b6c79c
MD
70};
71
93bed9fe
JG
72struct run_as_mkdir_data {
73 int dirfd;
74 char path[LTTNG_PATH_MAX];
75 mode_t mode;
76} LTTNG_PACKED;
77
e11d277b 78struct run_as_open_data {
93bed9fe
JG
79 int dirfd;
80 char path[LTTNG_PATH_MAX];
60b6c79c
MD
81 int flags;
82 mode_t mode;
93bed9fe 83} LTTNG_PACKED;
60b6c79c 84
4628484a 85struct run_as_unlink_data {
93bed9fe
JG
86 int dirfd;
87 char path[LTTNG_PATH_MAX];
88} LTTNG_PACKED;
4628484a 89
93bed9fe
JG
90struct run_as_rmdir_data {
91 int dirfd;
92 char path[LTTNG_PATH_MAX];
f75c5439 93 int flags; /* enum lttng_directory_handle_rmdir_recursive_flags */
93bed9fe 94} LTTNG_PACKED;
7567352f 95
241e0a5a 96struct run_as_extract_elf_symbol_offset_data {
93bed9fe 97 int fd;
241e0a5a 98 char function[LTTNG_SYMBOL_NAME_LEN];
93bed9fe 99} LTTNG_PACKED;
241e0a5a 100
0ef03255 101struct run_as_extract_sdt_probe_offsets_data {
93bed9fe 102 int fd;
0ef03255
FD
103 char probe_name[LTTNG_SYMBOL_NAME_LEN];
104 char provider_name[LTTNG_SYMBOL_NAME_LEN];
93bed9fe 105} LTTNG_PACKED;
0ef03255 106
93bed9fe
JG
107struct run_as_rename_data {
108 /*
109 * [0] = old_dirfd
110 * [1] = new_dirfd
111 */
112 int dirfds[2];
113 char old_path[LTTNG_PATH_MAX];
114 char new_path[LTTNG_PATH_MAX];
115} LTTNG_PACKED;
fe9f7760
FD
116
117struct run_as_open_ret {
93bed9fe
JG
118 int fd;
119} LTTNG_PACKED;
fe9f7760 120
241e0a5a
FD
121struct run_as_extract_elf_symbol_offset_ret {
122 uint64_t offset;
93bed9fe 123} LTTNG_PACKED;
241e0a5a 124
0ef03255
FD
125struct run_as_extract_sdt_probe_offsets_ret {
126 uint32_t num_offset;
127 uint64_t offsets[LTTNG_KERNEL_MAX_UPROBE_NUM];
93bed9fe 128} LTTNG_PACKED;
7567352f
MD
129
130struct run_as_data {
131 enum run_as_cmd cmd;
132 union {
93bed9fe 133 struct run_as_mkdir_data mkdir;
7567352f
MD
134 struct run_as_open_data open;
135 struct run_as_unlink_data unlink;
93bed9fe
JG
136 struct run_as_rmdir_data rmdir;
137 struct run_as_rename_data rename;
241e0a5a 138 struct run_as_extract_elf_symbol_offset_data extract_elf_symbol_offset;
0ef03255 139 struct run_as_extract_sdt_probe_offsets_data extract_sdt_probe_offsets;
7567352f
MD
140 } u;
141 uid_t uid;
142 gid_t gid;
93bed9fe 143} LTTNG_PACKED;
4628484a 144
fe9f7760
FD
145/*
146 * The run_as_ret structure holds the returned value and status of the command.
147 *
148 * The `u` union field holds the return value of the command; in most cases it
149 * represents the success or the failure of the command. In more complex
150 * commands, it holds a computed value.
151 *
152 * The _errno field is the errno recorded after the execution of the command.
153 *
154 * The _error fields is used the signify that return status of the command. For
155 * simple commands returning `int` the _error field will be the same as the
156 * ret_int field. In complex commands, it signify the success or failure of the
157 * command.
158 *
159 */
df5b86c8 160struct run_as_ret {
fe9f7760 161 union {
93bed9fe 162 int ret;
fe9f7760 163 struct run_as_open_ret open;
241e0a5a 164 struct run_as_extract_elf_symbol_offset_ret extract_elf_symbol_offset;
0ef03255 165 struct run_as_extract_sdt_probe_offsets_ret extract_sdt_probe_offsets;
fe9f7760 166 } u;
df5b86c8 167 int _errno;
fe9f7760 168 bool _error;
93bed9fe
JG
169} LTTNG_PACKED;
170
171#define COMMAND_IN_FDS(data_ptr) ({ \
172 int *fds = NULL; \
173 if (command_properties[data_ptr->cmd].in_fds_offset != -1) { \
174 fds = (int *) ((char *) data_ptr + command_properties[data_ptr->cmd].in_fds_offset); \
175 } \
176 fds; \
177})
178
179#define COMMAND_OUT_FDS(cmd, ret_ptr) ({ \
180 int *fds = NULL; \
181 if (command_properties[cmd].out_fds_offset != -1) { \
182 fds = (int *) ((char *) ret_ptr + command_properties[cmd].out_fds_offset); \
183 } \
184 fds; \
185})
186
187#define COMMAND_IN_FD_COUNT(data_ptr) ({ \
188 command_properties[data_ptr->cmd].in_fd_count; \
189})
190
191#define COMMAND_OUT_FD_COUNT(cmd) ({ \
192 command_properties[cmd].out_fd_count; \
193})
194
195#define COMMAND_USE_CWD_FD(data_ptr) command_properties[data_ptr->cmd].use_cwd_fd
196
197struct run_as_command_properties {
198 /* Set to -1 when not applicable. */
199 ptrdiff_t in_fds_offset, out_fds_offset;
200 unsigned int in_fd_count, out_fd_count;
201 bool use_cwd_fd;
202};
203
204static const struct run_as_command_properties command_properties[] = {
205 [RUN_AS_MKDIR] = {
206 .in_fds_offset = offsetof(struct run_as_data, u.mkdir.dirfd),
207 .in_fd_count = 1,
208 .out_fds_offset = -1,
209 .out_fd_count = 0,
210 .use_cwd_fd = true,
211 },
212 [RUN_AS_MKDIRAT] = {
213 .in_fds_offset = offsetof(struct run_as_data, u.mkdir.dirfd),
214 .in_fd_count = 1,
215 .out_fds_offset = -1,
216 .out_fd_count = 0,
217 .use_cwd_fd = false,
218 },
219 [RUN_AS_MKDIR_RECURSIVE] = {
220 .in_fds_offset = offsetof(struct run_as_data, u.mkdir.dirfd),
221 .in_fd_count = 1,
222 .out_fds_offset = -1,
223 .out_fd_count = 0,
224 .use_cwd_fd = true,
225 },
226 [RUN_AS_MKDIRAT_RECURSIVE] = {
227 .in_fds_offset = offsetof(struct run_as_data, u.mkdir.dirfd),
228 .in_fd_count = 1,
229 .out_fds_offset = -1,
230 .out_fd_count = 0,
231 .use_cwd_fd = false,
232 },
233 [RUN_AS_OPEN] = {
234 .in_fds_offset = offsetof(struct run_as_data, u.open.dirfd),
235 .in_fd_count = 1,
236 .out_fds_offset = offsetof(struct run_as_ret, u.open.fd),
237 .out_fd_count = 1,
238 .use_cwd_fd = true,
239 },
240 [RUN_AS_OPENAT] = {
241 .in_fds_offset = offsetof(struct run_as_data, u.open.dirfd),
242 .in_fd_count = 1,
243 .out_fds_offset = offsetof(struct run_as_ret, u.open.fd),
244 .out_fd_count = 1,
245 .use_cwd_fd = false,
246 },
247 [RUN_AS_UNLINK] = {
248 .in_fds_offset = offsetof(struct run_as_data, u.unlink.dirfd),
249 .in_fd_count = 1,
250 .out_fds_offset = -1,
251 .out_fd_count = 0,
252 .use_cwd_fd = true,
253 },
254 [RUN_AS_UNLINKAT] = {
255 .in_fds_offset = offsetof(struct run_as_data, u.unlink.dirfd),
256 .in_fd_count = 1,
257 .out_fds_offset = -1,
258 .out_fd_count = 0,
259 .use_cwd_fd = false,
260 },
261 [RUN_AS_RMDIR_RECURSIVE] = {
262 .in_fds_offset = offsetof(struct run_as_data, u.rmdir.dirfd),
263 .in_fd_count = 1,
264 .out_fds_offset = -1,
265 .out_fd_count = 0,
266 .use_cwd_fd = true,
267 },
268 [RUN_AS_RMDIRAT_RECURSIVE] = {
269 .in_fds_offset = offsetof(struct run_as_data, u.rmdir.dirfd),
270 .in_fd_count = 1,
271 .out_fds_offset = -1,
272 .out_fd_count = 0,
273 .use_cwd_fd = false,
274 },
275 [RUN_AS_RMDIR] = {
276 .in_fds_offset = offsetof(struct run_as_data, u.rmdir.dirfd),
277 .in_fd_count = 1,
278 .out_fds_offset = -1,
279 .out_fd_count = 0,
280 .use_cwd_fd = true,
281 },
282 [RUN_AS_RMDIRAT] = {
283 .in_fds_offset = offsetof(struct run_as_data, u.rmdir.dirfd),
284 .in_fd_count = 1,
285 .out_fds_offset = -1,
286 .out_fd_count = 0,
287 .use_cwd_fd = false,
288 },
289 [RUN_AS_RENAME] = {
290 .in_fds_offset = offsetof(struct run_as_data, u.rename.dirfds),
291 .in_fd_count = 2,
292 .out_fds_offset = -1,
293 .out_fd_count = 0,
294 .use_cwd_fd = true,
295 },
296 [RUN_AS_RENAMEAT] = {
297 .in_fds_offset = offsetof(struct run_as_data, u.rename.dirfds),
298 .in_fd_count = 2,
299 .out_fds_offset = -1,
300 .out_fd_count = 0,
301 .use_cwd_fd = false,
302 },
303 [RUN_AS_EXTRACT_ELF_SYMBOL_OFFSET] = {
304 .in_fds_offset = offsetof(struct run_as_data,
305 u.extract_elf_symbol_offset.fd),
306 .in_fd_count = 1,
307 .out_fds_offset = -1,
308 .out_fd_count = 0,
309 .use_cwd_fd = false,
310 },
311 [RUN_AS_EXTRACT_SDT_PROBE_OFFSETS] = {
312 .in_fds_offset = offsetof(struct run_as_data,
313 u.extract_sdt_probe_offsets.fd),
314 .in_fd_count = 1,
315 .out_fds_offset = -1,
316 .out_fd_count = 0,
317 .use_cwd_fd = false,
318 },
df5b86c8
MD
319};
320
7567352f
MD
321struct run_as_worker {
322 pid_t pid; /* Worker PID. */
323 int sockpair[2];
324 char *procname;
325};
326
327/* Single global worker per process (for now). */
328static struct run_as_worker *global_worker;
329/* Lock protecting the worker. */
330static pthread_mutex_t worker_lock = PTHREAD_MUTEX_INITIALIZER;
331
8f0044bf
MD
332#ifdef VALGRIND
333static
334int use_clone(void)
335{
336 return 0;
337}
338#else
339static
340int use_clone(void)
341{
e8fa9fb0 342 return !lttng_secure_getenv("LTTNG_DEBUG_NOCLONE");
8f0044bf
MD
343}
344#endif
345
60b6c79c
MD
346/*
347 * Create recursively directory using the FULL path.
348 */
349static
18710679 350int _mkdirat_recursive(struct run_as_data *data, struct run_as_ret *ret_value)
60b6c79c 351{
60b6c79c 352 const char *path;
60b6c79c 353 mode_t mode;
cbf53d23 354 struct lttng_directory_handle *handle;
60b6c79c 355
93bed9fe
JG
356 path = data->u.mkdir.path;
357 mode = data->u.mkdir.mode;
60b6c79c 358
cbf53d23
JG
359 handle = lttng_directory_handle_create_from_dirfd(data->u.mkdir.dirfd);
360 if (!handle) {
361 ret_value->_errno = errno;
362 ret_value->_error = true;
363 ret_value->u.ret = -1;
364 goto end;
365 }
87bbb856 366 /* Ownership of dirfd is transferred to the handle. */
93bed9fe 367 data->u.mkdir.dirfd = -1;
d77dded2 368 /* Safe to call as we have transitioned to the requested uid/gid. */
cbf53d23
JG
369 ret_value->u.ret = lttng_directory_handle_create_subdirectory_recursive(
370 handle, path, mode);
fe9f7760 371 ret_value->_errno = errno;
93bed9fe 372 ret_value->_error = (ret_value->u.ret) ? true : false;
cbf53d23
JG
373 lttng_directory_handle_put(handle);
374end:
93bed9fe 375 return ret_value->u.ret;
60b6c79c
MD
376}
377
378static
18710679 379int _mkdirat(struct run_as_data *data, struct run_as_ret *ret_value)
60b6c79c 380{
18710679
JG
381 const char *path;
382 mode_t mode;
cbf53d23 383 struct lttng_directory_handle *handle;
18710679 384
93bed9fe
JG
385 path = data->u.mkdir.path;
386 mode = data->u.mkdir.mode;
18710679 387
cbf53d23
JG
388 handle = lttng_directory_handle_create_from_dirfd(data->u.mkdir.dirfd);
389 if (!handle) {
390 ret_value->u.ret = -1;
391 ret_value->_errno = errno;
392 ret_value->_error = true;
393 goto end;
394 }
87bbb856 395 /* Ownership of dirfd is transferred to the handle. */
93bed9fe 396 data->u.mkdir.dirfd = -1;
18710679 397 /* Safe to call as we have transitioned to the requested uid/gid. */
cbf53d23
JG
398 ret_value->u.ret = lttng_directory_handle_create_subdirectory(
399 handle, path, mode);
fe9f7760 400 ret_value->_errno = errno;
93bed9fe 401 ret_value->_error = (ret_value->u.ret) ? true : false;
cbf53d23
JG
402 lttng_directory_handle_put(handle);
403end:
93bed9fe 404 return ret_value->u.ret;
7567352f 405}
7ce36756 406
7567352f 407static
fe9f7760 408int _open(struct run_as_data *data, struct run_as_ret *ret_value)
7567352f 409{
93bed9fe 410 int fd;
cbf53d23 411 struct lttng_directory_handle *handle;
93bed9fe 412
cbf53d23
JG
413 handle = lttng_directory_handle_create_from_dirfd(data->u.open.dirfd);
414 if (!handle) {
415 ret_value->_errno = errno;
416 ret_value->_error = true;
417 ret_value->u.ret = -1;
418 goto end;
419 }
93bed9fe
JG
420 /* Ownership of dirfd is transferred to the handle. */
421 data->u.open.dirfd = -1;
422
cbf53d23 423 fd = lttng_directory_handle_open_file(handle,
93bed9fe
JG
424 data->u.open.path, data->u.open.flags,
425 data->u.open.mode);
426 if (fd < 0) {
427 ret_value->u.ret = -1;
428 ret_value->u.open.fd = -1;
429 } else {
430 ret_value->u.ret = 0;
431 ret_value->u.open.fd = fd;
432 }
433
fe9f7760 434 ret_value->_errno = errno;
93bed9fe 435 ret_value->_error = fd < 0;
cbf53d23
JG
436 lttng_directory_handle_put(handle);
437end:
93bed9fe 438 return ret_value->u.ret;
7567352f
MD
439}
440
441static
fe9f7760 442int _unlink(struct run_as_data *data, struct run_as_ret *ret_value)
7567352f 443{
cbf53d23 444 struct lttng_directory_handle *handle;
93bed9fe 445
cbf53d23
JG
446 handle = lttng_directory_handle_create_from_dirfd(data->u.unlink.dirfd);
447 if (!handle) {
448 ret_value->u.ret = -1;
449 ret_value->_errno = errno;
450 ret_value->_error = true;
451 goto end;
452 }
93bed9fe
JG
453
454 /* Ownership of dirfd is transferred to the handle. */
455 data->u.unlink.dirfd = -1;
456
cbf53d23 457 ret_value->u.ret = lttng_directory_handle_unlink_file(handle,
93bed9fe
JG
458 data->u.unlink.path);
459 ret_value->_errno = errno;
460 ret_value->_error = (ret_value->u.ret) ? true : false;
cbf53d23
JG
461 lttng_directory_handle_put(handle);
462end:
93bed9fe
JG
463 return ret_value->u.ret;
464}
465
466static
467int _rmdir(struct run_as_data *data, struct run_as_ret *ret_value)
468{
cbf53d23 469 struct lttng_directory_handle *handle;
93bed9fe 470
cbf53d23
JG
471 handle = lttng_directory_handle_create_from_dirfd(data->u.rmdir.dirfd);
472 if (!handle) {
473 ret_value->u.ret = -1;
474 ret_value->_errno = errno;
475 ret_value->_error = true;
476 goto end;
477 }
93bed9fe
JG
478
479 /* Ownership of dirfd is transferred to the handle. */
480 data->u.rmdir.dirfd = -1;
481
482 ret_value->u.ret = lttng_directory_handle_remove_subdirectory(
cbf53d23 483 handle, data->u.rmdir.path);
fe9f7760 484 ret_value->_errno = errno;
93bed9fe 485 ret_value->_error = (ret_value->u.ret) ? true : false;
cbf53d23
JG
486 lttng_directory_handle_put(handle);
487end:
93bed9fe 488 return ret_value->u.ret;
60b6c79c
MD
489}
490
491static
fe9f7760 492int _rmdir_recursive(struct run_as_data *data, struct run_as_ret *ret_value)
60b6c79c 493{
cbf53d23 494 struct lttng_directory_handle *handle;
93bed9fe 495
cbf53d23
JG
496 handle = lttng_directory_handle_create_from_dirfd(data->u.rmdir.dirfd);
497 if (!handle) {
498 ret_value->u.ret = -1;
499 ret_value->_errno = errno;
500 ret_value->_error = true;
501 goto end;
502 }
93bed9fe
JG
503
504 /* Ownership of dirfd is transferred to the handle. */
505 data->u.rmdir.dirfd = -1;
506
507 ret_value->u.ret = lttng_directory_handle_remove_subdirectory_recursive(
cbf53d23 508 handle, data->u.rmdir.path, data->u.rmdir.flags);
fe9f7760 509 ret_value->_errno = errno;
93bed9fe 510 ret_value->_error = (ret_value->u.ret) ? true : false;
cbf53d23
JG
511 lttng_directory_handle_put(handle);
512end:
93bed9fe
JG
513 return ret_value->u.ret;
514}
515
516static
517int _rename(struct run_as_data *data, struct run_as_ret *ret_value)
518{
519 const char *old_path, *new_path;
cbf53d23 520 struct lttng_directory_handle *old_handle = NULL, *new_handle = NULL;
93bed9fe
JG
521
522 old_path = data->u.rename.old_path;
523 new_path = data->u.rename.new_path;
524
cbf53d23 525 old_handle = lttng_directory_handle_create_from_dirfd(
93bed9fe 526 data->u.rename.dirfds[0]);
cbf53d23
JG
527 if (!old_handle) {
528 ret_value->u.ret = -1;
529 goto end;
530 }
531 new_handle = lttng_directory_handle_create_from_dirfd(
93bed9fe 532 data->u.rename.dirfds[1]);
cbf53d23
JG
533 if (!new_handle) {
534 ret_value->u.ret = -1;
535 goto end;
536 }
93bed9fe
JG
537
538 /* Ownership of dirfds are transferred to the handles. */
539 data->u.rename.dirfds[0] = data->u.rename.dirfds[1] = -1;
540
541 /* Safe to call as we have transitioned to the requested uid/gid. */
542 ret_value->u.ret = lttng_directory_handle_rename(
cbf53d23
JG
543 old_handle, old_path, new_handle, new_path);
544end:
545 lttng_directory_handle_put(old_handle);
546 lttng_directory_handle_put(new_handle);
93bed9fe
JG
547 ret_value->_errno = errno;
548 ret_value->_error = (ret_value->u.ret) ? true : false;
93bed9fe 549 return ret_value->u.ret;
7567352f 550}
df5b86c8 551
b1b34226 552#ifdef HAVE_ELF_H
241e0a5a
FD
553static
554int _extract_elf_symbol_offset(struct run_as_data *data,
555 struct run_as_ret *ret_value)
556{
557 int ret = 0;
65eefd4c 558 uint64_t offset;
241e0a5a 559
93bed9fe
JG
560 ret_value->_error = false;
561 ret = lttng_elf_get_symbol_offset(data->u.extract_elf_symbol_offset.fd,
241e0a5a 562 data->u.extract_elf_symbol_offset.function,
65eefd4c 563 &offset);
241e0a5a
FD
564 if (ret) {
565 DBG("Failed to extract ELF function offset");
566 ret_value->_error = true;
567 }
65eefd4c 568 ret_value->u.extract_elf_symbol_offset.offset = offset;
241e0a5a
FD
569
570 return ret;
571}
572
0ef03255
FD
573static
574int _extract_sdt_probe_offsets(struct run_as_data *data,
575 struct run_as_ret *ret_value)
576{
577 int ret = 0;
578 uint64_t *offsets = NULL;
579 uint32_t num_offset;
580
581 ret_value->_error = false;
582
583 /* On success, this call allocates the offsets paramater. */
93bed9fe
JG
584 ret = lttng_elf_get_sdt_probe_offsets(
585 data->u.extract_sdt_probe_offsets.fd,
0ef03255
FD
586 data->u.extract_sdt_probe_offsets.provider_name,
587 data->u.extract_sdt_probe_offsets.probe_name,
588 &offsets, &num_offset);
589
590 if (ret) {
591 DBG("Failed to extract SDT probe offsets");
592 ret_value->_error = true;
593 goto end;
594 }
595
596 if (num_offset <= 0 || num_offset > LTTNG_KERNEL_MAX_UPROBE_NUM) {
597 DBG("Wrong number of probes.");
598 ret = -1;
599 ret_value->_error = true;
600 goto free_offset;
601 }
602
603 /* Copy the content of the offsets array to the ret struct. */
604 memcpy(ret_value->u.extract_sdt_probe_offsets.offsets,
605 offsets, num_offset * sizeof(uint64_t));
606
607 ret_value->u.extract_sdt_probe_offsets.num_offset = num_offset;
608
609free_offset:
610 free(offsets);
611end:
612 return ret;
613}
b1b34226
MJ
614#else
615static
616int _extract_elf_symbol_offset(struct run_as_data *data,
617 struct run_as_ret *ret_value)
618{
619 ERR("Unimplemented runas command RUN_AS_EXTRACT_ELF_SYMBOL_OFFSET");
620 return -1;
621}
622
623static
624int _extract_sdt_probe_offsets(struct run_as_data *data,
625 struct run_as_ret *ret_value)
626{
627 ERR("Unimplemented runas command RUN_AS_EXTRACT_SDT_PROBE_OFFSETS");
628 return -1;
629}
630#endif
241e0a5a 631
7567352f
MD
632static
633run_as_fct run_as_enum_to_fct(enum run_as_cmd cmd)
634{
635 switch (cmd) {
636 case RUN_AS_MKDIR:
18710679
JG
637 case RUN_AS_MKDIRAT:
638 return _mkdirat;
639 case RUN_AS_MKDIR_RECURSIVE:
640 case RUN_AS_MKDIRAT_RECURSIVE:
641 return _mkdirat_recursive;
7567352f 642 case RUN_AS_OPEN:
2912cead 643 case RUN_AS_OPENAT:
7567352f
MD
644 return _open;
645 case RUN_AS_UNLINK:
2912cead 646 case RUN_AS_UNLINKAT:
7567352f 647 return _unlink;
93bed9fe
JG
648 case RUN_AS_RMDIR:
649 case RUN_AS_RMDIRAT:
650 return _rmdir;
7567352f 651 case RUN_AS_RMDIR_RECURSIVE:
93bed9fe 652 case RUN_AS_RMDIRAT_RECURSIVE:
7567352f 653 return _rmdir_recursive;
93bed9fe
JG
654 case RUN_AS_RENAME:
655 case RUN_AS_RENAMEAT:
656 return _rename;
241e0a5a
FD
657 case RUN_AS_EXTRACT_ELF_SYMBOL_OFFSET:
658 return _extract_elf_symbol_offset;
0ef03255
FD
659 case RUN_AS_EXTRACT_SDT_PROBE_OFFSETS:
660 return _extract_sdt_probe_offsets;
7567352f 661 default:
62a7b8ed 662 ERR("Unknown command %d", (int) cmd);
7567352f
MD
663 return NULL;
664 }
60b6c79c
MD
665}
666
4628484a 667static
93bed9fe 668int do_send_fds(int sock, const int *fds, unsigned int fd_count)
4628484a 669{
7567352f 670 ssize_t len;
93bed9fe
JG
671 unsigned int i;
672
673 for (i = 0; i < fd_count; i++) {
674 if (fds[i] < 0) {
675 ERR("Attempt to send invalid file descriptor to master (fd = %i)",
676 fds[i]);
677 /* Return 0 as this is not a fatal error. */
678 return 0;
679 }
680 }
4628484a 681
93bed9fe
JG
682 len = lttcomm_send_fds_unix_sock(sock, fds, fd_count);
683 return len < 0 ? -1 : 0;
4628484a
MD
684}
685
686static
93bed9fe 687int do_recv_fds(int sock, int *fds, unsigned int fd_count)
4628484a 688{
93bed9fe
JG
689 int ret = 0;
690 unsigned int i;
7567352f 691 ssize_t len;
4628484a 692
93bed9fe
JG
693 len = lttcomm_recv_fds_unix_sock(sock, fds, fd_count);
694 if (len == 0) {
695 ret = -1;
696 goto end;
da9ee832 697 } else if (len < 0) {
93bed9fe
JG
698 PERROR("Failed to receive file descriptors from socket");
699 ret = -1;
700 goto end;
ca9eb994
JG
701 }
702
93bed9fe
JG
703 for (i = 0; i < fd_count; i++) {
704 if (fds[i] < 0) {
705 ERR("Invalid file descriptor received from worker (fd = %i)", fds[i]);
706 /* Return 0 as this is not a fatal error. */
707 }
708 }
709end:
710 return ret;
4628484a
MD
711}
712
fe9f7760 713static
93bed9fe
JG
714int send_fds_to_worker(const struct run_as_worker *worker,
715 const struct run_as_data *data)
fe9f7760
FD
716{
717 int ret = 0;
93bed9fe 718 unsigned int i;
fe9f7760 719
93bed9fe
JG
720 if (COMMAND_USE_CWD_FD(data) || COMMAND_IN_FD_COUNT(data) == 0) {
721 goto end;
fe9f7760
FD
722 }
723
93bed9fe
JG
724 for (i = 0; i < COMMAND_IN_FD_COUNT(data); i++) {
725 if (COMMAND_IN_FDS(data)[i] < 0) {
726 ERR("Refusing to send invalid fd to worker (fd = %i)",
727 COMMAND_IN_FDS(data)[i]);
728 ret = -1;
729 goto end;
730 }
731 }
ca9eb994 732
93bed9fe
JG
733 ret = do_send_fds(worker->sockpair[0], COMMAND_IN_FDS(data),
734 COMMAND_IN_FD_COUNT(data));
fe9f7760 735 if (ret < 0) {
93bed9fe 736 PERROR("Failed to send file descriptor to run-as worker");
fe9f7760 737 ret = -1;
93bed9fe 738 goto end;
fe9f7760 739 }
93bed9fe 740end:
fe9f7760
FD
741 return ret;
742}
743
744static
93bed9fe
JG
745int send_fds_to_master(struct run_as_worker *worker, enum run_as_cmd cmd,
746 struct run_as_ret *run_as_ret)
fe9f7760 747{
93bed9fe
JG
748 int ret = 0;
749 unsigned int i;
fe9f7760 750
93bed9fe
JG
751 if (COMMAND_OUT_FD_COUNT(cmd) == 0) {
752 goto end;
fe9f7760
FD
753 }
754
93bed9fe
JG
755 ret = do_send_fds(worker->sockpair[1], COMMAND_OUT_FDS(cmd, run_as_ret),
756 COMMAND_OUT_FD_COUNT(cmd));
fe9f7760 757 if (ret < 0) {
93bed9fe
JG
758 PERROR("Failed to send file descriptor to master process");
759 goto end;
fe9f7760
FD
760 }
761
93bed9fe
JG
762 for (i = 0; i < COMMAND_OUT_FD_COUNT(cmd); i++) {
763 int ret_close = close(COMMAND_OUT_FDS(cmd, run_as_ret)[i]);
ca9eb994 764
93bed9fe
JG
765 if (ret_close < 0) {
766 PERROR("Failed to close result file descriptor");
767 }
768 }
769end:
fe9f7760
FD
770 return ret;
771}
772
773static
93bed9fe
JG
774int recv_fds_from_worker(const struct run_as_worker *worker, enum run_as_cmd cmd,
775 struct run_as_ret *run_as_ret)
fe9f7760
FD
776{
777 int ret = 0;
778
93bed9fe
JG
779 if (COMMAND_OUT_FD_COUNT(cmd) == 0) {
780 goto end;
fe9f7760
FD
781 }
782
93bed9fe
JG
783 ret = do_recv_fds(worker->sockpair[0], COMMAND_OUT_FDS(cmd, run_as_ret),
784 COMMAND_OUT_FD_COUNT(cmd));
fe9f7760 785 if (ret < 0) {
93bed9fe 786 PERROR("Failed to receive file descriptor from run-as worker");
fe9f7760
FD
787 ret = -1;
788 }
93bed9fe 789end:
fe9f7760
FD
790 return ret;
791}
792
793static
93bed9fe 794int recv_fds_from_master(struct run_as_worker *worker, struct run_as_data *data)
fe9f7760
FD
795{
796 int ret = 0;
797
93bed9fe
JG
798 if (COMMAND_USE_CWD_FD(data)) {
799 unsigned int i;
800
801 for (i = 0; i < COMMAND_IN_FD_COUNT(data); i++) {
802 COMMAND_IN_FDS(data)[i] = AT_FDCWD;
803 }
804 goto end;
fe9f7760
FD
805 }
806
93bed9fe
JG
807 ret = do_recv_fds(worker->sockpair[1], COMMAND_IN_FDS(data),
808 COMMAND_IN_FD_COUNT(data));
fe9f7760 809 if (ret < 0) {
93bed9fe 810 PERROR("Failed to receive file descriptors from master process");
fe9f7760
FD
811 ret = -1;
812 }
93bed9fe 813end:
fe9f7760
FD
814 return ret;
815}
816
817static
93bed9fe 818int cleanup_received_fds(struct run_as_data *data)
fe9f7760 819{
93bed9fe 820 int ret = 0, i;
fe9f7760 821
93bed9fe
JG
822 for (i = 0; i < COMMAND_IN_FD_COUNT(data); i++) {
823 if (COMMAND_IN_FDS(data)[i] == -1) {
824 continue;
825 }
826 ret = close(COMMAND_IN_FDS(data)[i]);
827 if (ret) {
828 PERROR("Failed to close file descriptor received fd in run-as worker");
829 goto end;
830 }
fe9f7760 831 }
93bed9fe 832end:
fe9f7760
FD
833 return ret;
834}
0ef03255 835
7567352f
MD
836/*
837 * Return < 0 on error, 0 if OK, 1 on hangup.
838 */
c2b75c49 839static
7567352f 840int handle_one_cmd(struct run_as_worker *worker)
c2b75c49 841{
7567352f 842 int ret = 0;
93bed9fe
JG
843 struct run_as_data data = {};
844 ssize_t readlen, writelen;
845 struct run_as_ret sendret = {};
846 run_as_fct cmd;
7567352f
MD
847 uid_t prev_euid;
848
fe9f7760
FD
849 /*
850 * Stage 1: Receive run_as_data struct from the master.
851 * The structure contains the command type and all the parameters needed for
852 * its execution
853 */
7567352f
MD
854 readlen = lttcomm_recv_unix_sock(worker->sockpair[1], &data,
855 sizeof(data));
856 if (readlen == 0) {
857 /* hang up */
858 ret = 1;
859 goto end;
860 }
861 if (readlen < sizeof(data)) {
862 PERROR("lttcomm_recv_unix_sock error");
863 ret = -1;
864 goto end;
865 }
c2b75c49 866
7567352f
MD
867 cmd = run_as_enum_to_fct(data.cmd);
868 if (!cmd) {
869 ret = -1;
870 goto end;
871 }
872
fe9f7760
FD
873 /*
874 * Stage 2: Receive file descriptor from master.
875 * Some commands need a file descriptor as input so if it's needed we
876 * receive the fd using the Unix socket.
877 */
93bed9fe 878 ret = recv_fds_from_master(worker, &data);
fe9f7760
FD
879 if (ret < 0) {
880 PERROR("recv_fd_from_master error");
881 ret = -1;
882 goto end;
883 }
884
7567352f
MD
885 prev_euid = getuid();
886 if (data.gid != getegid()) {
887 ret = setegid(data.gid);
1576d582 888 if (ret < 0) {
561f5f2c
JG
889 sendret._error = true;
890 sendret._errno = errno;
4c462e79 891 PERROR("setegid");
6d73c4ef 892 goto write_return;
1576d582 893 }
c2b75c49 894 }
7567352f
MD
895 if (data.uid != prev_euid) {
896 ret = seteuid(data.uid);
1576d582 897 if (ret < 0) {
561f5f2c
JG
898 sendret._error = true;
899 sendret._errno = errno;
4c462e79 900 PERROR("seteuid");
6d73c4ef 901 goto write_return;
1576d582 902 }
c2b75c49 903 }
fe9f7760 904
c2b75c49
MD
905 /*
906 * Also set umask to 0 for mkdir executable bit.
907 */
908 umask(0);
fe9f7760
FD
909
910 /*
911 * Stage 3: Execute the command
912 */
913 ret = (*cmd)(&data, &sendret);
914 if (ret < 0) {
915 DBG("Execution of command returned an error");
916 }
6d73c4ef
MD
917
918write_return:
93bed9fe 919 ret = cleanup_received_fds(&data);
fe9f7760
FD
920 if (ret < 0) {
921 ERR("Error cleaning up FD");
922 goto end;
923 }
924
925 /*
926 * Stage 4: Send run_as_ret structure to the master.
927 * This structure contain the return value of the command and the errno.
928 */
7567352f
MD
929 writelen = lttcomm_send_unix_sock(worker->sockpair[1], &sendret,
930 sizeof(sendret));
6cd525e8 931 if (writelen < sizeof(sendret)) {
7567352f
MD
932 PERROR("lttcomm_send_unix_sock error");
933 ret = -1;
934 goto end;
935 }
fe9f7760
FD
936
937 /*
93bed9fe 938 * Stage 5: Send resulting file descriptors to the master.
fe9f7760 939 */
93bed9fe 940 ret = send_fds_to_master(worker, data.cmd, &sendret);
fe9f7760
FD
941 if (ret < 0) {
942 DBG("Sending FD to master returned an error");
7567352f
MD
943 goto end;
944 }
fe9f7760 945
7567352f
MD
946 if (seteuid(prev_euid) < 0) {
947 PERROR("seteuid");
948 ret = -1;
949 goto end;
950 }
951 ret = 0;
952end:
953 return ret;
954}
955
956static
957int run_as_worker(struct run_as_worker *worker)
958{
959 int ret;
960 ssize_t writelen;
961 struct run_as_ret sendret;
962 size_t proc_orig_len;
963
964 /*
965 * Initialize worker. Set a different process cmdline.
966 */
967 proc_orig_len = strlen(worker->procname);
968 memset(worker->procname, 0, proc_orig_len);
969 strncpy(worker->procname, DEFAULT_RUN_AS_WORKER_NAME, proc_orig_len);
970
e1055edb
JG
971 ret = lttng_prctl(PR_SET_NAME,
972 (unsigned long) DEFAULT_RUN_AS_WORKER_NAME, 0, 0, 0);
973 if (ret && ret != -ENOSYS) {
b8090274
JG
974 /* Don't fail as this is not essential. */
975 PERROR("prctl PR_SET_NAME");
6cd525e8 976 }
7567352f 977
fe9f7760
FD
978 memset(&sendret, 0, sizeof(sendret));
979
7567352f
MD
980 writelen = lttcomm_send_unix_sock(worker->sockpair[1], &sendret,
981 sizeof(sendret));
982 if (writelen < sizeof(sendret)) {
983 PERROR("lttcomm_send_unix_sock error");
984 ret = EXIT_FAILURE;
985 goto end;
986 }
987
988 for (;;) {
989 ret = handle_one_cmd(worker);
990 if (ret < 0) {
991 ret = EXIT_FAILURE;
992 goto end;
993 } else if (ret > 0) {
994 break;
995 } else {
996 continue; /* Next command. */
997 }
998 }
999 ret = EXIT_SUCCESS;
1000end:
1001 return ret;
c2b75c49
MD
1002}
1003
60b6c79c 1004static
7567352f
MD
1005int run_as_cmd(struct run_as_worker *worker,
1006 enum run_as_cmd cmd,
1007 struct run_as_data *data,
fe9f7760 1008 struct run_as_ret *ret_value,
7567352f 1009 uid_t uid, gid_t gid)
60b6c79c 1010{
fe9f7760 1011 int ret = 0;
7567352f 1012 ssize_t readlen, writelen;
60b6c79c
MD
1013
1014 /*
1015 * If we are non-root, we can only deal with our own uid.
1016 */
1017 if (geteuid() != 0) {
1018 if (uid != geteuid()) {
fe9f7760
FD
1019 ret = -1;
1020 ret_value->_errno = EPERM;
60b6c79c 1021 ERR("Client (%d)/Server (%d) UID mismatch (and sessiond is not root)",
08797918 1022 (int) uid, (int) geteuid());
df5b86c8 1023 goto end;
60b6c79c 1024 }
60b6c79c
MD
1025 }
1026
7567352f
MD
1027 data->cmd = cmd;
1028 data->uid = uid;
1029 data->gid = gid;
1030
fe9f7760
FD
1031 /*
1032 * Stage 1: Send the run_as_data struct to the worker process
1033 */
7567352f
MD
1034 writelen = lttcomm_send_unix_sock(worker->sockpair[0], data,
1035 sizeof(*data));
1036 if (writelen < sizeof(*data)) {
1037 PERROR("Error writing message to run_as");
fe9f7760
FD
1038 ret = -1;
1039 ret_value->_errno = EIO;
60b6c79c 1040 goto end;
c2b75c49 1041 }
7567352f 1042
fe9f7760
FD
1043 /*
1044 * Stage 2: Send file descriptor to the worker process if needed
1045 */
93bed9fe 1046 ret = send_fds_to_worker(worker, data);
fe9f7760
FD
1047 if (ret) {
1048 PERROR("do_send_fd error");
1049 ret = -1;
1050 ret_value->_errno = EIO;
1051 goto end;
1052 }
1053
1054 /*
1055 * Stage 3: Wait for the execution of the command
1056 */
1057
1058 /*
1059 * Stage 4: Receive the run_as_ret struct containing the return value and
1060 * errno
1061 */
1062 readlen = lttcomm_recv_unix_sock(worker->sockpair[0], ret_value,
1063 sizeof(*ret_value));
da9ee832
JG
1064 if (!readlen) {
1065 ERR("Run-as worker has hung-up during run_as_cmd");
fe9f7760
FD
1066 ret = -1;
1067 ret_value->_errno = EIO;
da9ee832 1068 goto end;
fe9f7760 1069 } else if (readlen < sizeof(*ret_value)) {
7567352f 1070 PERROR("Error reading response from run_as");
fe9f7760
FD
1071 ret = -1;
1072 ret_value->_errno = errno;
033b58a7 1073 goto end;
6cd525e8 1074 }
fe9f7760 1075
ca9eb994
JG
1076 if (ret_value->_error) {
1077 /* Skip stage 5 on error as there will be no fd to receive. */
1078 goto end;
1079 }
1080
fe9f7760
FD
1081 /*
1082 * Stage 5: Receive file descriptor if needed
1083 */
93bed9fe 1084 ret = recv_fds_from_worker(worker, cmd, ret_value);
fe9f7760
FD
1085 if (ret < 0) {
1086 ERR("Error receiving fd");
1087 ret = -1;
1088 ret_value->_errno = EIO;
4c462e79 1089 }
7567352f 1090
60b6c79c 1091end:
fe9f7760 1092 return ret;
60b6c79c
MD
1093}
1094
2d85a600 1095/*
7567352f 1096 * This is for debugging ONLY and should not be considered secure.
2d85a600
MD
1097 */
1098static
fe9f7760
FD
1099int run_as_noworker(enum run_as_cmd cmd,
1100 struct run_as_data *data, struct run_as_ret *ret_value,
1101 uid_t uid, gid_t gid)
2d85a600 1102{
df5b86c8 1103 int ret, saved_errno;
5b73926f 1104 mode_t old_mask;
7567352f 1105 run_as_fct fct;
5b73926f 1106
7567352f
MD
1107 fct = run_as_enum_to_fct(cmd);
1108 if (!fct) {
1109 errno = -ENOSYS;
1110 ret = -1;
1111 goto end;
1112 }
5b73926f 1113 old_mask = umask(0);
fe9f7760
FD
1114 ret = fct(data, ret_value);
1115 saved_errno = ret_value->_errno;
5b73926f 1116 umask(old_mask);
df5b86c8 1117 errno = saved_errno;
7567352f 1118end:
5b73926f 1119 return ret;
2d85a600
MD
1120}
1121
8fec83ea
JG
1122static
1123int reset_sighandler(void)
1124{
1125 int sig;
1126
1127 DBG("Resetting run_as worker signal handlers to default");
1128 for (sig = 1; sig <= 31; sig++) {
1129 (void) signal(sig, SIG_DFL);
1130 }
1131 return 0;
1132}
1133
1134static
1135void worker_sighandler(int sig)
1136{
1137 const char *signame;
1138
1139 /*
1140 * The worker will inherit its parent's signals since they are part of
1141 * the same process group. However, in the case of SIGINT and SIGTERM,
1142 * we want to give the worker a chance to teardown gracefully when its
1143 * parent closes the command socket.
1144 */
1145 switch (sig) {
1146 case SIGINT:
1147 signame = "SIGINT";
1148 break;
1149 case SIGTERM:
1150 signame = "SIGTERM";
1151 break;
1152 default:
1153 signame = NULL;
1154 }
1155
1156 if (signame) {
1157 DBG("run_as worker received signal %s", signame);
1158 } else {
1159 DBG("run_as_worker received signal %d", sig);
1160 }
1161}
1162
1163static
1164int set_worker_sighandlers(void)
1165{
1166 int ret = 0;
1167 sigset_t sigset;
1168 struct sigaction sa;
1169
1170 if ((ret = sigemptyset(&sigset)) < 0) {
1171 PERROR("sigemptyset");
1172 goto end;
1173 }
1174
1175 sa.sa_handler = worker_sighandler;
1176 sa.sa_mask = sigset;
1177 sa.sa_flags = 0;
1178 if ((ret = sigaction(SIGINT, &sa, NULL)) < 0) {
1179 PERROR("sigaction SIGINT");
1180 goto end;
1181 }
1182
1183 if ((ret = sigaction(SIGTERM, &sa, NULL)) < 0) {
1184 PERROR("sigaction SIGTERM");
1185 goto end;
1186 }
1187
1188 DBG("run_as signal handler set for SIGTERM and SIGINT");
1189end:
1190 return ret;
1191}
1192
1193static
929f71ec
JG
1194int run_as_create_worker_no_lock(const char *procname,
1195 post_fork_cleanup_cb clean_up_func,
1196 void *clean_up_user_data)
8fec83ea
JG
1197{
1198 pid_t pid;
1199 int i, ret = 0;
1200 ssize_t readlen;
1201 struct run_as_ret recvret;
1202 struct run_as_worker *worker;
1203
1204 assert(!global_worker);
1205 if (!use_clone()) {
1206 /*
1207 * Don't initialize a worker, all run_as tasks will be performed
1208 * in the current process.
1209 */
1210 ret = 0;
1211 goto end;
1212 }
1213 worker = zmalloc(sizeof(*worker));
1214 if (!worker) {
1215 ret = -ENOMEM;
1216 goto end;
1217 }
1218 worker->procname = strdup(procname);
1219 if (!worker->procname) {
1220 ret = -ENOMEM;
8c96eded 1221 goto error_procname_alloc;
8fec83ea
JG
1222 }
1223 /* Create unix socket. */
1224 if (lttcomm_create_anon_unix_socketpair(worker->sockpair) < 0) {
1225 ret = -1;
1226 goto error_sock;
1227 }
1228
1229 /* Fork worker. */
1230 pid = fork();
1231 if (pid < 0) {
1232 PERROR("fork");
1233 ret = -1;
1234 goto error_fork;
1235 } else if (pid == 0) {
1236 /* Child */
1237
1238 reset_sighandler();
1239
1240 set_worker_sighandlers();
929f71ec
JG
1241 if (clean_up_func) {
1242 if (clean_up_func(clean_up_user_data) < 0) {
1243 ERR("Run-as post-fork clean-up failed, exiting.");
1244 exit(EXIT_FAILURE);
1245 }
1246 }
8fec83ea
JG
1247
1248 /* Just close, no shutdown. */
1249 if (close(worker->sockpair[0])) {
1250 PERROR("close");
1251 exit(EXIT_FAILURE);
1252 }
1253
1254 /*
1255 * Close all FDs aside from STDIN, STDOUT, STDERR and sockpair[1]
1256 * Sockpair[1] is used as a control channel with the master
1257 */
1258 for (i = 3; i < sysconf(_SC_OPEN_MAX); i++) {
1259 if (i != worker->sockpair[1]) {
1260 (void) close(i);
1261 }
1262 }
1263
1264 worker->sockpair[0] = -1;
1265 ret = run_as_worker(worker);
1266 if (lttcomm_close_unix_sock(worker->sockpair[1])) {
1267 PERROR("close");
1268 ret = -1;
1269 }
1270 worker->sockpair[1] = -1;
340cf672
JG
1271 free(worker->procname);
1272 free(worker);
8fec83ea
JG
1273 LOG(ret ? PRINT_ERR : PRINT_DBG, "run_as worker exiting (ret = %d)", ret);
1274 exit(ret ? EXIT_FAILURE : EXIT_SUCCESS);
1275 } else {
1276 /* Parent */
1277
1278 /* Just close, no shutdown. */
1279 if (close(worker->sockpair[1])) {
1280 PERROR("close");
1281 ret = -1;
1282 goto error_fork;
1283 }
1284 worker->sockpair[1] = -1;
1285 worker->pid = pid;
1286 /* Wait for worker to become ready. */
1287 readlen = lttcomm_recv_unix_sock(worker->sockpair[0],
1288 &recvret, sizeof(recvret));
1289 if (readlen < sizeof(recvret)) {
1290 ERR("readlen: %zd", readlen);
1291 PERROR("Error reading response from run_as at creation");
1292 ret = -1;
1293 goto error_fork;
1294 }
1295 global_worker = worker;
1296 }
1297end:
1298 return ret;
1299
1300 /* Error handling. */
1301error_fork:
1302 for (i = 0; i < 2; i++) {
1303 if (worker->sockpair[i] < 0) {
1304 continue;
1305 }
1306 if (lttcomm_close_unix_sock(worker->sockpair[i])) {
1307 PERROR("close");
1308 }
1309 worker->sockpair[i] = -1;
1310 }
1311error_sock:
8c96eded
FD
1312 free(worker->procname);
1313error_procname_alloc:
8fec83ea
JG
1314 free(worker);
1315 return ret;
1316}
1317
a01c682b
JR
1318static
1319void run_as_destroy_worker_no_lock(void)
1320{
1321 struct run_as_worker *worker = global_worker;
1322
1323 DBG("Destroying run_as worker");
1324 if (!worker) {
1325 return;
1326 }
1327 /* Close unix socket */
1328 DBG("Closing run_as worker socket");
1329 if (lttcomm_close_unix_sock(worker->sockpair[0])) {
1330 PERROR("close");
1331 }
1332 worker->sockpair[0] = -1;
1333 /* Wait for worker. */
1334 for (;;) {
1335 int status;
1336 pid_t wait_ret;
1337
1338 wait_ret = waitpid(worker->pid, &status, 0);
1339 if (wait_ret < 0) {
1340 if (errno == EINTR) {
1341 continue;
1342 }
1343 PERROR("waitpid");
1344 break;
1345 }
1346
1347 if (WIFEXITED(status)) {
1348 LOG(WEXITSTATUS(status) == 0 ? PRINT_DBG : PRINT_ERR,
1349 DEFAULT_RUN_AS_WORKER_NAME " terminated with status code %d",
1350 WEXITSTATUS(status));
1351 break;
1352 } else if (WIFSIGNALED(status)) {
1353 ERR(DEFAULT_RUN_AS_WORKER_NAME " was killed by signal %d",
1354 WTERMSIG(status));
1355 break;
1356 }
1357 }
1358 free(worker->procname);
1359 free(worker);
1360 global_worker = NULL;
1361}
1362
2d85a600 1363static
fe9f7760 1364int run_as_restart_worker(struct run_as_worker *worker)
2d85a600 1365{
fe9f7760
FD
1366 int ret = 0;
1367 char *procname = NULL;
1368
1369 procname = worker->procname;
1370
1371 /* Close socket to run_as worker process and clean up the zombie process */
a01c682b 1372 run_as_destroy_worker_no_lock();
fe9f7760
FD
1373
1374 /* Create a new run_as worker process*/
929f71ec 1375 ret = run_as_create_worker_no_lock(procname, NULL, NULL);
fe9f7760
FD
1376 if (ret < 0 ) {
1377 ERR("Restarting the worker process failed");
1378 ret = -1;
1379 goto err;
1380 }
1381err:
1382 return ret;
1383}
1384
1385static
1386int run_as(enum run_as_cmd cmd, struct run_as_data *data,
1387 struct run_as_ret *ret_value, uid_t uid, gid_t gid)
1388{
1389 int ret, saved_errno;
7567352f 1390
8fec83ea 1391 pthread_mutex_lock(&worker_lock);
749b7a0c 1392 if (use_clone()) {
7567352f 1393 DBG("Using run_as worker");
8fec83ea 1394
749b7a0c 1395 assert(global_worker);
749b7a0c 1396
fe9f7760
FD
1397 ret = run_as_cmd(global_worker, cmd, data, ret_value, uid, gid);
1398 saved_errno = ret_value->_errno;
1399
fe9f7760
FD
1400 /*
1401 * If the worker thread crashed the errno is set to EIO. we log
1402 * the error and start a new worker process.
1403 */
1404 if (ret == -1 && saved_errno == EIO) {
1405 DBG("Socket closed unexpectedly... "
1406 "Restarting the worker process");
1407 ret = run_as_restart_worker(global_worker);
fe9f7760
FD
1408 if (ret == -1) {
1409 ERR("Failed to restart worker process.");
1410 goto err;
1411 }
1412 }
2d85a600 1413 } else {
7567352f 1414 DBG("Using run_as without worker");
fe9f7760 1415 ret = run_as_noworker(cmd, data, ret_value, uid, gid);
2d85a600 1416 }
fe9f7760 1417err:
8fec83ea 1418 pthread_mutex_unlock(&worker_lock);
7567352f 1419 return ret;
2d85a600
MD
1420}
1421
90e535ef 1422LTTNG_HIDDEN
e11d277b 1423int run_as_mkdir_recursive(const char *path, mode_t mode, uid_t uid, gid_t gid)
60b6c79c 1424{
18710679
JG
1425 return run_as_mkdirat_recursive(AT_FDCWD, path, mode, uid, gid);
1426}
1427
1428LTTNG_HIDDEN
1429int run_as_mkdirat_recursive(int dirfd, const char *path, mode_t mode,
1430 uid_t uid, gid_t gid)
1431{
1432 int ret;
93bed9fe
JG
1433 struct run_as_data data = {};
1434 struct run_as_ret run_as_ret = {};
60b6c79c 1435
18710679
JG
1436 DBG3("mkdirat() recursive fd = %d%s, path = %s, mode = %d, uid = %d, gid = %d",
1437 dirfd, dirfd == AT_FDCWD ? " (AT_FDCWD)" : "",
08797918 1438 path, (int) mode, (int) uid, (int) gid);
93bed9fe
JG
1439 ret = lttng_strncpy(data.u.mkdir.path, path,
1440 sizeof(data.u.mkdir.path));
18710679
JG
1441 if (ret) {
1442 ERR("Failed to copy path argument of mkdirat recursive command");
1443 goto error;
1444 }
93bed9fe
JG
1445 data.u.mkdir.path[sizeof(data.u.mkdir.path) - 1] = '\0';
1446 data.u.mkdir.mode = mode;
1447 data.u.mkdir.dirfd = dirfd;
18710679
JG
1448 run_as(dirfd == AT_FDCWD ? RUN_AS_MKDIR_RECURSIVE : RUN_AS_MKDIRAT_RECURSIVE,
1449 &data, &run_as_ret, uid, gid);
1450 errno = run_as_ret._errno;
93bed9fe 1451 ret = run_as_ret.u.ret;
18710679
JG
1452error:
1453 return ret;
60b6c79c
MD
1454}
1455
90e535ef 1456LTTNG_HIDDEN
e11d277b 1457int run_as_mkdir(const char *path, mode_t mode, uid_t uid, gid_t gid)
60b6c79c 1458{
18710679
JG
1459 return run_as_mkdirat(AT_FDCWD, path, mode, uid, gid);
1460}
1461
1462LTTNG_HIDDEN
1463int run_as_mkdirat(int dirfd, const char *path, mode_t mode,
1464 uid_t uid, gid_t gid)
1465{
1466 int ret;
93bed9fe
JG
1467 struct run_as_data data = {};
1468 struct run_as_ret run_as_ret = {};
fe9f7760 1469
18710679
JG
1470 DBG3("mkdirat() recursive fd = %d%s, path = %s, mode = %d, uid = %d, gid = %d",
1471 dirfd, dirfd == AT_FDCWD ? " (AT_FDCWD)" : "",
08797918 1472 path, (int) mode, (int) uid, (int) gid);
93bed9fe
JG
1473 ret = lttng_strncpy(data.u.mkdir.path, path,
1474 sizeof(data.u.mkdir.path));
18710679
JG
1475 if (ret) {
1476 ERR("Failed to copy path argument of mkdirat command");
1477 goto error;
1478 }
93bed9fe
JG
1479 data.u.mkdir.path[sizeof(data.u.mkdir.path) - 1] = '\0';
1480 data.u.mkdir.mode = mode;
1481 data.u.mkdir.dirfd = dirfd;
18710679
JG
1482 run_as(dirfd == AT_FDCWD ? RUN_AS_MKDIR : RUN_AS_MKDIRAT,
1483 &data, &run_as_ret, uid, gid);
1484 errno = run_as_ret._errno;
93bed9fe 1485 ret = run_as_ret.u.ret;
18710679
JG
1486error:
1487 return ret;
60b6c79c
MD
1488}
1489
90e535ef 1490LTTNG_HIDDEN
2912cead
JG
1491int run_as_open(const char *path, int flags, mode_t mode, uid_t uid,
1492 gid_t gid)
1493{
1494 return run_as_openat(AT_FDCWD, path, flags, mode, uid, gid);
1495}
1496
1497LTTNG_HIDDEN
1498int run_as_openat(int dirfd, const char *path, int flags, mode_t mode,
1499 uid_t uid, gid_t gid)
60b6c79c 1500{
93bed9fe
JG
1501 int ret;
1502 struct run_as_data data = {};
1503 struct run_as_ret run_as_ret = {};
fe9f7760 1504
2912cead
JG
1505 DBG3("openat() fd = %d%s, path = %s, flags = %X, mode = %d, uid %d, gid %d",
1506 dirfd, dirfd == AT_FDCWD ? " (AT_FDCWD)" : "",
08797918 1507 path, flags, (int) mode, (int) uid, (int) gid);
93bed9fe
JG
1508 ret = lttng_strncpy(data.u.open.path, path, sizeof(data.u.open.path));
1509 if (ret) {
1510 ERR("Failed to copy path argument of open command");
1511 goto error;
1512 }
7567352f
MD
1513 data.u.open.flags = flags;
1514 data.u.open.mode = mode;
93bed9fe 1515 data.u.open.dirfd = dirfd;
2912cead 1516 run_as(dirfd == AT_FDCWD ? RUN_AS_OPEN : RUN_AS_OPENAT,
93bed9fe
JG
1517 &data, &run_as_ret, uid, gid);
1518 errno = run_as_ret._errno;
1519 ret = run_as_ret.u.ret < 0 ? run_as_ret.u.ret :
1520 run_as_ret.u.open.fd;
1521error:
1522 return ret;
60b6c79c 1523}
4628484a
MD
1524
1525LTTNG_HIDDEN
1526int run_as_unlink(const char *path, uid_t uid, gid_t gid)
2912cead
JG
1527{
1528 return run_as_unlinkat(AT_FDCWD, path, uid, gid);
1529}
1530
1531LTTNG_HIDDEN
1532int run_as_unlinkat(int dirfd, const char *path, uid_t uid, gid_t gid)
4628484a 1533{
93bed9fe
JG
1534 int ret;
1535 struct run_as_data data = {};
1536 struct run_as_ret run_as_ret = {};
fe9f7760 1537
2912cead
JG
1538 DBG3("unlinkat() fd = %d%s, path = %s, uid = %d, gid = %d",
1539 dirfd, dirfd == AT_FDCWD ? " (AT_FDCWD)" : "",
08797918 1540 path, (int) uid, (int) gid);
93bed9fe
JG
1541 ret = lttng_strncpy(data.u.unlink.path, path,
1542 sizeof(data.u.unlink.path));
1543 if (ret) {
1544 goto error;
1545 }
1546 data.u.unlink.dirfd = dirfd;
1547 run_as(dirfd == AT_FDCWD ? RUN_AS_UNLINK : RUN_AS_UNLINKAT, &data,
1548 &run_as_ret, uid, gid);
1549 errno = run_as_ret._errno;
1550 ret = run_as_ret.u.ret;
1551error:
1552 return ret;
1553}
1554
1555LTTNG_HIDDEN
1556int run_as_rmdir(const char *path, uid_t uid, gid_t gid)
1557{
1558 return run_as_rmdirat(AT_FDCWD, path, uid, gid);
1559}
1560
1561LTTNG_HIDDEN
1562int run_as_rmdirat(int dirfd, const char *path, uid_t uid, gid_t gid)
1563{
1564 int ret;
1565 struct run_as_data data = {};
1566 struct run_as_ret run_as_ret = {};
1567
1568 DBG3("rmdirat() fd = %d%s, path = %s, uid = %d, gid = %d",
1569 dirfd, dirfd == AT_FDCWD ? " (AT_FDCWD)" : "",
1570 path, (int) uid, (int) gid);
1571 ret = lttng_strncpy(data.u.rmdir.path, path,
1572 sizeof(data.u.rmdir.path));
1573 if (ret) {
1574 goto error;
1575 }
1576 data.u.rmdir.dirfd = dirfd;
1577 run_as(dirfd == AT_FDCWD ? RUN_AS_RMDIR : RUN_AS_RMDIRAT, &data,
1578 &run_as_ret, uid, gid);
1579 errno = run_as_ret._errno;
1580 ret = run_as_ret.u.ret;
1581error:
1582 return ret;
4628484a
MD
1583}
1584
1585LTTNG_HIDDEN
f75c5439 1586int run_as_rmdir_recursive(const char *path, uid_t uid, gid_t gid, int flags)
4628484a 1587{
f75c5439 1588 return run_as_rmdirat_recursive(AT_FDCWD, path, uid, gid, flags);
93bed9fe 1589}
fe9f7760 1590
93bed9fe 1591LTTNG_HIDDEN
f75c5439 1592int run_as_rmdirat_recursive(int dirfd, const char *path, uid_t uid, gid_t gid, int flags)
93bed9fe
JG
1593{
1594 int ret;
1595 struct run_as_data data = {};
1596 struct run_as_ret run_as_ret = {};
4628484a 1597
93bed9fe
JG
1598 DBG3("rmdirat() recursive fd = %d%s, path = %s, uid = %d, gid = %d",
1599 dirfd, dirfd == AT_FDCWD ? " (AT_FDCWD)" : "",
08797918 1600 path, (int) uid, (int) gid);
93bed9fe
JG
1601 ret = lttng_strncpy(data.u.rmdir.path, path,
1602 sizeof(data.u.rmdir.path));
1603 if (ret) {
1604 goto error;
1605 }
1606 data.u.rmdir.dirfd = dirfd;
f75c5439 1607 data.u.rmdir.flags = flags;
93bed9fe
JG
1608 run_as(dirfd == AT_FDCWD ? RUN_AS_RMDIR_RECURSIVE : RUN_AS_RMDIRAT_RECURSIVE,
1609 &data, &run_as_ret, uid, gid);
1610 errno = run_as_ret._errno;
1611 ret = run_as_ret.u.ret;
1612error:
1613 return ret;
1614}
1615
1616LTTNG_HIDDEN
1617int run_as_rename(const char *old, const char *new, uid_t uid, gid_t gid)
1618{
1619 return run_as_renameat(AT_FDCWD, old, AT_FDCWD, new, uid, gid);
1620}
1621
1622LTTNG_HIDDEN
1623int run_as_renameat(int old_dirfd, const char *old_name,
1624 int new_dirfd, const char *new_name, uid_t uid, gid_t gid)
1625{
1626 int ret;
1627 struct run_as_data data = {};
1628 struct run_as_ret run_as_ret = {};
1629
1630 DBG3("renameat() old_dirfd = %d%s, old_name = %s, new_dirfd = %d%s, new_name = %s, uid = %d, gid = %d",
1631 old_dirfd, old_dirfd == AT_FDCWD ? " (AT_FDCWD)" : "",
1632 old_name,
1633 new_dirfd, new_dirfd == AT_FDCWD ? " (AT_FDCWD)" : "",
1634 new_name, (int) uid, (int) gid);
1635 ret = lttng_strncpy(data.u.rename.old_path, old_name,
1636 sizeof(data.u.rename.old_path));
1637 if (ret) {
1638 goto error;
1639 }
1640 ret = lttng_strncpy(data.u.rename.new_path, new_name,
1641 sizeof(data.u.rename.new_path));
1642 if (ret) {
1643 goto error;
1644 }
1645
1646 data.u.rename.dirfds[0] = old_dirfd;
1647 data.u.rename.dirfds[1] = new_dirfd;
1648 run_as(old_dirfd == AT_FDCWD && new_dirfd == AT_FDCWD ?
1649 RUN_AS_RENAME : RUN_AS_RENAMEAT,
1650 &data, &run_as_ret, uid, gid);
1651 errno = run_as_ret._errno;
1652 ret = run_as_ret.u.ret;
1653error:
1654 return ret;
7567352f
MD
1655}
1656
241e0a5a
FD
1657LTTNG_HIDDEN
1658int run_as_extract_elf_symbol_offset(int fd, const char* function,
1659 uid_t uid, gid_t gid, uint64_t *offset)
1660{
93bed9fe
JG
1661 int ret;
1662 struct run_as_data data = {};
1663 struct run_as_ret run_as_ret = {};
f726677b 1664
241e0a5a 1665 DBG3("extract_elf_symbol_offset() on fd=%d and function=%s "
93bed9fe
JG
1666 "with for uid %d and gid %d", fd, function,
1667 (int) uid, (int) gid);
241e0a5a 1668
93bed9fe 1669 data.u.extract_elf_symbol_offset.fd = fd;
241e0a5a
FD
1670
1671 strncpy(data.u.extract_elf_symbol_offset.function, function, LTTNG_SYMBOL_NAME_LEN - 1);
241e0a5a 1672 data.u.extract_elf_symbol_offset.function[LTTNG_SYMBOL_NAME_LEN - 1] = '\0';
93bed9fe
JG
1673 ret = lttng_strncpy(data.u.extract_elf_symbol_offset.function,
1674 function,
1675 sizeof(data.u.extract_elf_symbol_offset.function));
1676 if (ret) {
1677 goto error;
1678 }
241e0a5a 1679
93bed9fe
JG
1680 run_as(RUN_AS_EXTRACT_ELF_SYMBOL_OFFSET, &data, &run_as_ret, uid, gid);
1681 errno = run_as_ret._errno;
1682 if (run_as_ret._error) {
1683 ret = -1;
1684 goto error;
241e0a5a
FD
1685 }
1686
93bed9fe
JG
1687 *offset = run_as_ret.u.extract_elf_symbol_offset.offset;
1688error:
1689 return ret;
241e0a5a
FD
1690}
1691
0ef03255
FD
1692LTTNG_HIDDEN
1693int run_as_extract_sdt_probe_offsets(int fd, const char* provider_name,
1694 const char* probe_name, uid_t uid, gid_t gid,
1695 uint64_t **offsets, uint32_t *num_offset)
1696{
93bed9fe
JG
1697 int ret;
1698 struct run_as_data data = {};
1699 struct run_as_ret run_as_ret = {};
f726677b 1700
0ef03255 1701 DBG3("extract_sdt_probe_offsets() on fd=%d, probe_name=%s and "
93bed9fe
JG
1702 "provider_name=%s with for uid %d and gid %d", fd,
1703 probe_name, provider_name, (int) uid, (int) gid);
0ef03255 1704
93bed9fe 1705 data.u.extract_sdt_probe_offsets.fd = fd;
0ef03255 1706
93bed9fe
JG
1707 ret = lttng_strncpy(data.u.extract_sdt_probe_offsets.probe_name, probe_name,
1708 sizeof(data.u.extract_sdt_probe_offsets.probe_name));
1709 if (ret) {
1710 goto error;
1711 }
1712 ret = lttng_strncpy(data.u.extract_sdt_probe_offsets.provider_name,
1713 provider_name,
1714 sizeof(data.u.extract_sdt_probe_offsets.provider_name));
1715 if (ret) {
1716 goto error;
0ef03255
FD
1717 }
1718
93bed9fe
JG
1719 run_as(RUN_AS_EXTRACT_SDT_PROBE_OFFSETS, &data, &run_as_ret, uid, gid);
1720 errno = run_as_ret._errno;
1721 if (run_as_ret._error) {
1722 ret = -1;
1723 goto error;
1724 }
0ef03255 1725
93bed9fe 1726 *num_offset = run_as_ret.u.extract_sdt_probe_offsets.num_offset;
0ef03255
FD
1727 *offsets = zmalloc(*num_offset * sizeof(uint64_t));
1728 if (!*offsets) {
93bed9fe
JG
1729 ret = -ENOMEM;
1730 goto error;
0ef03255
FD
1731 }
1732
93bed9fe
JG
1733 memcpy(*offsets, run_as_ret.u.extract_sdt_probe_offsets.offsets,
1734 *num_offset * sizeof(uint64_t));
1735error:
1736 return ret;
0ef03255
FD
1737}
1738
7567352f 1739LTTNG_HIDDEN
929f71ec
JG
1740int run_as_create_worker(const char *procname,
1741 post_fork_cleanup_cb clean_up_func,
1742 void *clean_up_user_data)
7567352f 1743{
8fec83ea 1744 int ret;
7567352f 1745
749b7a0c 1746 pthread_mutex_lock(&worker_lock);
929f71ec
JG
1747 ret = run_as_create_worker_no_lock(procname, clean_up_func,
1748 clean_up_user_data);
749b7a0c 1749 pthread_mutex_unlock(&worker_lock);
7567352f
MD
1750 return ret;
1751}
1752
1753LTTNG_HIDDEN
1754void run_as_destroy_worker(void)
1755{
749b7a0c 1756 pthread_mutex_lock(&worker_lock);
a01c682b 1757 run_as_destroy_worker_no_lock();
749b7a0c 1758 pthread_mutex_unlock(&worker_lock);
4628484a 1759}
This page took 0.132707 seconds and 4 git commands to generate.