1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
|
/*
* Copyright (c) 2019-2021 Dmitry V. Levin <ldv@strace.io>
* All rights reserved.
*
* SPDX-License-Identifier: LGPL-2.1-or-later
*/
#include "defs.h"
#include <linux/fcntl.h>
#include <linux/mount.h>
#include "xlat/mount_setattr_flags.h"
#include "xlat/mount_attr_attr.h"
#include "xlat/mount_attr_propagation.h"
static void
print_mount_attr(struct tcb *const tcp,
const kernel_ulong_t addr,
const kernel_ulong_t size)
{
struct mount_attr attr;
if (size < MOUNT_ATTR_SIZE_VER0) {
printaddr(addr);
return;
}
if (umoven_or_printaddr(tcp, addr, MIN(sizeof(attr), size), &attr))
return;
tprint_struct_begin();
PRINT_FIELD_FLAGS(attr, attr_set, mount_attr_attr, "MOUNT_ATTR_???");
tprint_struct_next();
PRINT_FIELD_FLAGS(attr, attr_clr, mount_attr_attr, "MOUNT_ATTR_???");
tprint_struct_next();
PRINT_FIELD_XVAL(attr, propagation, mount_attr_propagation, "MS_???");
tprint_struct_next();
if (attr.userns_fd > INT_MAX ||
!((attr.attr_set | attr.attr_clr) & MOUNT_ATTR_IDMAP)) {
PRINT_FIELD_U(attr, userns_fd);
} else {
PRINT_FIELD_FD(attr, userns_fd, tcp);
}
if (size > sizeof(attr)) {
print_nonzero_bytes(tcp, tprint_struct_next, addr, sizeof(attr),
MIN(size, get_pagesize()), QUOTE_FORCE_HEX);
}
tprint_struct_end();
}
SYS_FUNC(mount_setattr)
{
/* dirfd */
print_dirfd(tcp, tcp->u_arg[0]);
tprint_arg_next();
/* pathname */
printpath(tcp, tcp->u_arg[1]);
tprint_arg_next();
/* flags */
printflags(mount_setattr_flags, tcp->u_arg[2], "AT_???");
tprint_arg_next();
/* uattr */
print_mount_attr(tcp, tcp->u_arg[3], tcp->u_arg[4]);
tprint_arg_next();
/* usize */
PRINT_VAL_U(tcp->u_arg[4]);
return RVAL_DECODED | RVAL_FD;
}
|