1*49cdfc7eSAndroid Build Coastguard Worker /*
2*49cdfc7eSAndroid Build Coastguard Worker *
3*49cdfc7eSAndroid Build Coastguard Worker * Copyright (c) International Business Machines Corp., 2009
4*49cdfc7eSAndroid Build Coastguard Worker * Copyright (c) 2014 Oracle and/or its affiliates. All Rights Reserved.
5*49cdfc7eSAndroid Build Coastguard Worker *
6*49cdfc7eSAndroid Build Coastguard Worker * This program is free software; you can redistribute it and/or modify
7*49cdfc7eSAndroid Build Coastguard Worker * it under the terms of the GNU General Public License as published by
8*49cdfc7eSAndroid Build Coastguard Worker * the Free Software Foundation; either version 2 of the License, or
9*49cdfc7eSAndroid Build Coastguard Worker * (at your option) any later version.
10*49cdfc7eSAndroid Build Coastguard Worker *
11*49cdfc7eSAndroid Build Coastguard Worker * This program is distributed in the hope that it will be useful,
12*49cdfc7eSAndroid Build Coastguard Worker * but WITHOUT ANY WARRANTY; without even the implied warranty of
13*49cdfc7eSAndroid Build Coastguard Worker * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
14*49cdfc7eSAndroid Build Coastguard Worker * the GNU General Public License for more details.
15*49cdfc7eSAndroid Build Coastguard Worker *
16*49cdfc7eSAndroid Build Coastguard Worker * You should have received a copy of the GNU General Public License
17*49cdfc7eSAndroid Build Coastguard Worker * along with this program; if not, write to the Free Software
18*49cdfc7eSAndroid Build Coastguard Worker * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19*49cdfc7eSAndroid Build Coastguard Worker */
20*49cdfc7eSAndroid Build Coastguard Worker
21*49cdfc7eSAndroid Build Coastguard Worker #include <ctype.h>
22*49cdfc7eSAndroid Build Coastguard Worker #include <errno.h>
23*49cdfc7eSAndroid Build Coastguard Worker #include <fcntl.h>
24*49cdfc7eSAndroid Build Coastguard Worker #include <limits.h>
25*49cdfc7eSAndroid Build Coastguard Worker #include <stdio.h>
26*49cdfc7eSAndroid Build Coastguard Worker #include <stdlib.h>
27*49cdfc7eSAndroid Build Coastguard Worker #include <sys/types.h>
28*49cdfc7eSAndroid Build Coastguard Worker #include <unistd.h>
29*49cdfc7eSAndroid Build Coastguard Worker #include "test.h"
30*49cdfc7eSAndroid Build Coastguard Worker #include "tst_pid.h"
31*49cdfc7eSAndroid Build Coastguard Worker #include "old_safe_file_ops.h"
32*49cdfc7eSAndroid Build Coastguard Worker #include "tst_safe_macros.h"
33*49cdfc7eSAndroid Build Coastguard Worker #include "lapi/syscalls.h"
34*49cdfc7eSAndroid Build Coastguard Worker
35*49cdfc7eSAndroid Build Coastguard Worker #define PID_MAX_PATH "/proc/sys/kernel/pid_max"
36*49cdfc7eSAndroid Build Coastguard Worker #define THREADS_MAX_PATH "/proc/sys/kernel/threads-max"
37*49cdfc7eSAndroid Build Coastguard Worker #define CGROUPS_V1_SLICE_FMT "/sys/fs/cgroup/pids/user.slice/user-%d.slice/pids.max"
38*49cdfc7eSAndroid Build Coastguard Worker #define CGROUPS_V2_SLICE_FMT "/sys/fs/cgroup/user.slice/user-%d.slice/pids.max"
39*49cdfc7eSAndroid Build Coastguard Worker /* Leave some available processes for the OS */
40*49cdfc7eSAndroid Build Coastguard Worker #define PIDS_RESERVE 50
41*49cdfc7eSAndroid Build Coastguard Worker
tst_get_unused_pid_(void (* cleanup_fn)(void))42*49cdfc7eSAndroid Build Coastguard Worker pid_t tst_get_unused_pid_(void (*cleanup_fn) (void))
43*49cdfc7eSAndroid Build Coastguard Worker {
44*49cdfc7eSAndroid Build Coastguard Worker pid_t pid;
45*49cdfc7eSAndroid Build Coastguard Worker
46*49cdfc7eSAndroid Build Coastguard Worker SAFE_FILE_SCANF(cleanup_fn, PID_MAX_PATH, "%d", &pid);
47*49cdfc7eSAndroid Build Coastguard Worker
48*49cdfc7eSAndroid Build Coastguard Worker return pid;
49*49cdfc7eSAndroid Build Coastguard Worker }
50*49cdfc7eSAndroid Build Coastguard Worker
51*49cdfc7eSAndroid Build Coastguard Worker /*
52*49cdfc7eSAndroid Build Coastguard Worker * Get the effective session UID - either one invoking current test via sudo
53*49cdfc7eSAndroid Build Coastguard Worker * or the real UID.
54*49cdfc7eSAndroid Build Coastguard Worker */
get_session_uid(void)55*49cdfc7eSAndroid Build Coastguard Worker static unsigned int get_session_uid(void)
56*49cdfc7eSAndroid Build Coastguard Worker {
57*49cdfc7eSAndroid Build Coastguard Worker const char *sudo_uid;
58*49cdfc7eSAndroid Build Coastguard Worker
59*49cdfc7eSAndroid Build Coastguard Worker sudo_uid = getenv("SUDO_UID");
60*49cdfc7eSAndroid Build Coastguard Worker if (sudo_uid) {
61*49cdfc7eSAndroid Build Coastguard Worker unsigned int real_uid;
62*49cdfc7eSAndroid Build Coastguard Worker int ret;
63*49cdfc7eSAndroid Build Coastguard Worker
64*49cdfc7eSAndroid Build Coastguard Worker ret = sscanf(sudo_uid, "%u", &real_uid);
65*49cdfc7eSAndroid Build Coastguard Worker if (ret == 1)
66*49cdfc7eSAndroid Build Coastguard Worker return real_uid;
67*49cdfc7eSAndroid Build Coastguard Worker }
68*49cdfc7eSAndroid Build Coastguard Worker
69*49cdfc7eSAndroid Build Coastguard Worker return getuid();
70*49cdfc7eSAndroid Build Coastguard Worker }
71*49cdfc7eSAndroid Build Coastguard Worker
read_session_pids_limit(const char * path_fmt,int uid,void (* cleanup_fn)(void))72*49cdfc7eSAndroid Build Coastguard Worker static int read_session_pids_limit(const char *path_fmt, int uid,
73*49cdfc7eSAndroid Build Coastguard Worker void (*cleanup_fn) (void))
74*49cdfc7eSAndroid Build Coastguard Worker {
75*49cdfc7eSAndroid Build Coastguard Worker int max_pids, ret;
76*49cdfc7eSAndroid Build Coastguard Worker char max_pid_value[100];
77*49cdfc7eSAndroid Build Coastguard Worker char path[PATH_MAX];
78*49cdfc7eSAndroid Build Coastguard Worker
79*49cdfc7eSAndroid Build Coastguard Worker ret = snprintf(path, sizeof(path), path_fmt, uid);
80*49cdfc7eSAndroid Build Coastguard Worker if (ret < 0 || (size_t)ret >= sizeof(path))
81*49cdfc7eSAndroid Build Coastguard Worker return -1;
82*49cdfc7eSAndroid Build Coastguard Worker
83*49cdfc7eSAndroid Build Coastguard Worker if (access(path, R_OK) != 0) {
84*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "Cannot read session user limits from '%s'", path);
85*49cdfc7eSAndroid Build Coastguard Worker return -1;
86*49cdfc7eSAndroid Build Coastguard Worker }
87*49cdfc7eSAndroid Build Coastguard Worker
88*49cdfc7eSAndroid Build Coastguard Worker SAFE_FILE_SCANF(cleanup_fn, path, "%s", max_pid_value);
89*49cdfc7eSAndroid Build Coastguard Worker if (!strcmp(max_pid_value, "max")) {
90*49cdfc7eSAndroid Build Coastguard Worker SAFE_FILE_SCANF(cleanup_fn, PID_MAX_PATH, "%d", &max_pids);
91*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "Found limit of processes %d (from %s=max)", max_pids, path);
92*49cdfc7eSAndroid Build Coastguard Worker } else {
93*49cdfc7eSAndroid Build Coastguard Worker max_pids = SAFE_STRTOL(max_pid_value, 0, INT_MAX);
94*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "Found limit of processes %d (from %s)", max_pids, path);
95*49cdfc7eSAndroid Build Coastguard Worker }
96*49cdfc7eSAndroid Build Coastguard Worker
97*49cdfc7eSAndroid Build Coastguard Worker return max_pids;
98*49cdfc7eSAndroid Build Coastguard Worker }
99*49cdfc7eSAndroid Build Coastguard Worker
get_session_pids_limit(void (* cleanup_fn)(void))100*49cdfc7eSAndroid Build Coastguard Worker static int get_session_pids_limit(void (*cleanup_fn) (void))
101*49cdfc7eSAndroid Build Coastguard Worker {
102*49cdfc7eSAndroid Build Coastguard Worker int max_pids, uid;
103*49cdfc7eSAndroid Build Coastguard Worker
104*49cdfc7eSAndroid Build Coastguard Worker uid = get_session_uid();
105*49cdfc7eSAndroid Build Coastguard Worker max_pids = read_session_pids_limit(CGROUPS_V2_SLICE_FMT, uid, cleanup_fn);
106*49cdfc7eSAndroid Build Coastguard Worker if (max_pids < 0)
107*49cdfc7eSAndroid Build Coastguard Worker max_pids = read_session_pids_limit(CGROUPS_V1_SLICE_FMT, uid,
108*49cdfc7eSAndroid Build Coastguard Worker cleanup_fn);
109*49cdfc7eSAndroid Build Coastguard Worker
110*49cdfc7eSAndroid Build Coastguard Worker if (max_pids < 0)
111*49cdfc7eSAndroid Build Coastguard Worker return -1;
112*49cdfc7eSAndroid Build Coastguard Worker
113*49cdfc7eSAndroid Build Coastguard Worker return max_pids;
114*49cdfc7eSAndroid Build Coastguard Worker }
115*49cdfc7eSAndroid Build Coastguard Worker
get_used_pids(void (* cleanup_fn)(void))116*49cdfc7eSAndroid Build Coastguard Worker static int get_used_pids(void (*cleanup_fn) (void))
117*49cdfc7eSAndroid Build Coastguard Worker {
118*49cdfc7eSAndroid Build Coastguard Worker DIR *dir_proc;
119*49cdfc7eSAndroid Build Coastguard Worker struct dirent *ent;
120*49cdfc7eSAndroid Build Coastguard Worker char status_path[PATH_MAX];
121*49cdfc7eSAndroid Build Coastguard Worker int used_threads, used_pids = 0;
122*49cdfc7eSAndroid Build Coastguard Worker
123*49cdfc7eSAndroid Build Coastguard Worker dir_proc = SAFE_OPENDIR("/proc");
124*49cdfc7eSAndroid Build Coastguard Worker
125*49cdfc7eSAndroid Build Coastguard Worker while ((ent = SAFE_READDIR(dir_proc))) {
126*49cdfc7eSAndroid Build Coastguard Worker if (isdigit(ent->d_name[0])) {
127*49cdfc7eSAndroid Build Coastguard Worker snprintf(status_path, sizeof(status_path), "/proc/%s/status", ent->d_name);
128*49cdfc7eSAndroid Build Coastguard Worker if (!FILE_LINES_SCANF(cleanup_fn, status_path, "Threads: %d", &used_threads))
129*49cdfc7eSAndroid Build Coastguard Worker used_pids += used_threads;
130*49cdfc7eSAndroid Build Coastguard Worker }
131*49cdfc7eSAndroid Build Coastguard Worker }
132*49cdfc7eSAndroid Build Coastguard Worker
133*49cdfc7eSAndroid Build Coastguard Worker SAFE_CLOSEDIR(dir_proc);
134*49cdfc7eSAndroid Build Coastguard Worker
135*49cdfc7eSAndroid Build Coastguard Worker return used_pids;
136*49cdfc7eSAndroid Build Coastguard Worker }
137*49cdfc7eSAndroid Build Coastguard Worker
tst_get_free_pids_(void (* cleanup_fn)(void))138*49cdfc7eSAndroid Build Coastguard Worker int tst_get_free_pids_(void (*cleanup_fn) (void))
139*49cdfc7eSAndroid Build Coastguard Worker {
140*49cdfc7eSAndroid Build Coastguard Worker int max_pids, max_session_pids, max_threads, used_pids = get_used_pids(cleanup_fn);
141*49cdfc7eSAndroid Build Coastguard Worker
142*49cdfc7eSAndroid Build Coastguard Worker SAFE_FILE_SCANF(cleanup_fn, PID_MAX_PATH, "%d", &max_pids);
143*49cdfc7eSAndroid Build Coastguard Worker SAFE_FILE_SCANF(cleanup_fn, THREADS_MAX_PATH, "%d", &max_threads);
144*49cdfc7eSAndroid Build Coastguard Worker max_pids = MIN(max_pids, max_threads);
145*49cdfc7eSAndroid Build Coastguard Worker
146*49cdfc7eSAndroid Build Coastguard Worker max_session_pids = get_session_pids_limit(cleanup_fn);
147*49cdfc7eSAndroid Build Coastguard Worker if ((max_session_pids > 0) && (max_session_pids < max_pids))
148*49cdfc7eSAndroid Build Coastguard Worker max_pids = max_session_pids;
149*49cdfc7eSAndroid Build Coastguard Worker
150*49cdfc7eSAndroid Build Coastguard Worker if (max_pids > PIDS_RESERVE)
151*49cdfc7eSAndroid Build Coastguard Worker max_pids -= PIDS_RESERVE;
152*49cdfc7eSAndroid Build Coastguard Worker else
153*49cdfc7eSAndroid Build Coastguard Worker max_pids = 0;
154*49cdfc7eSAndroid Build Coastguard Worker
155*49cdfc7eSAndroid Build Coastguard Worker /* max_pids contains the maximum PID + 1,
156*49cdfc7eSAndroid Build Coastguard Worker * used_pids contains used PIDs + 1,
157*49cdfc7eSAndroid Build Coastguard Worker * so this additional '1' is eliminated by the substraction */
158*49cdfc7eSAndroid Build Coastguard Worker if (used_pids >= max_pids) {
159*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TBROK, cleanup_fn, "No free pids");
160*49cdfc7eSAndroid Build Coastguard Worker return 0;
161*49cdfc7eSAndroid Build Coastguard Worker }
162*49cdfc7eSAndroid Build Coastguard Worker return max_pids - used_pids;
163*49cdfc7eSAndroid Build Coastguard Worker }
164*49cdfc7eSAndroid Build Coastguard Worker
tst_getpid(void)165*49cdfc7eSAndroid Build Coastguard Worker pid_t tst_getpid(void)
166*49cdfc7eSAndroid Build Coastguard Worker {
167*49cdfc7eSAndroid Build Coastguard Worker return syscall(SYS_getpid);
168*49cdfc7eSAndroid Build Coastguard Worker }
169*49cdfc7eSAndroid Build Coastguard Worker
tst_gettid(void)170*49cdfc7eSAndroid Build Coastguard Worker pid_t tst_gettid(void)
171*49cdfc7eSAndroid Build Coastguard Worker {
172*49cdfc7eSAndroid Build Coastguard Worker return syscall(SYS_gettid);
173*49cdfc7eSAndroid Build Coastguard Worker }
174