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
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
|
/*
* Copyright (c) 2002-2004 Roland McGrath <roland@redhat.com>
* Copyright (c) 2009-2018 Dmitry V. Levin <ldv@strace.io>
* Copyright (c) 2014-2022 The strace developers.
* All rights reserved.
*
* SPDX-License-Identifier: LGPL-2.1-or-later
*/
#include "defs.h"
#include <sched.h>
static unsigned int
get_cpuset_size(void)
{
static unsigned int cpuset_size;
if (!cpuset_size) {
/*
* If the cpuset size passed to sched_getaffinity is less
* than necessary to store the bitmask, the kernel does not
* look at the mask pointer and fails with EINVAL.
*
* If the cpuset size is large enough, the kernel fails with
* EFAULT on inaccessible mask pointers.
*
* This undocumented kernel feature can be used to probe
* the kernel and find out the minimal valid cpuset size
* without allocating any memory for the CPU affinity mask.
*/
cpuset_size = 128;
while (cpuset_size &&
sched_getaffinity(0, cpuset_size, NULL) == -1 &&
EINVAL == errno) {
cpuset_size <<= 1;
}
if (!cpuset_size)
cpuset_size = 128;
}
return cpuset_size;
}
void
print_affinitylist(struct tcb *const tcp, const kernel_ulong_t addr,
const unsigned int len)
{
const unsigned int max_size = get_cpuset_size();
const unsigned int umove_size = MIN(len, max_size);
const unsigned int size = ROUNDUP(umove_size, current_wordsize);
const unsigned int ncpu = size * 8;
void *cpu;
if (!verbose(tcp) || (exiting(tcp) && syserror(tcp)) ||
!addr) {
printaddr(addr);
return;
}
if (!len) {
tprint_array_begin();
tprint_array_end();
return;
}
if (!(cpu = calloc(size, 1))) {
printaddr(addr);
return;
}
if (!umoven_or_printaddr(tcp, addr, umove_size, cpu)) {
bool printed = false;
tprint_bitset_begin();
for (int i = 0;; i++) {
i = next_set_bit(cpu, i, ncpu);
if (i < 0)
break;
if (printed)
tprint_bitset_next();
else
printed = true;
PRINT_VAL_D(i);
}
if (size < len) {
if (printed)
tprint_bitset_next();
tprint_more_data_follows();
}
tprint_bitset_end();
}
free(cpu);
}
SYS_FUNC(sched_setaffinity)
{
/* pid */
const int pid = tcp->u_arg[0];
printpid(tcp, pid, PT_TGID);
tprint_arg_next();
/* cpusetsize */
const unsigned int len = tcp->u_arg[1];
PRINT_VAL_U(len);
tprint_arg_next();
/* mask */
print_affinitylist(tcp, tcp->u_arg[2], len);
return RVAL_DECODED;
}
SYS_FUNC(sched_getaffinity)
{
if (entering(tcp)) {
/* pid */
const int pid = tcp->u_arg[0];
printpid(tcp, pid, PT_TGID);
tprint_arg_next();
/* cpusetsize */
const unsigned int len = tcp->u_arg[1];
PRINT_VAL_U(len);
tprint_arg_next();
} else {
/* mask */
print_affinitylist(tcp, tcp->u_arg[2], tcp->u_rval);
}
return 0;
}
|