blob: 8d63a73133156b6fe6f0725c6a2be1eaf73da8bd [file] [log] [blame]
Bernhard Reutner-Fischere15d7572006-06-02 20:56:16 +00001/* vi: set sw=4 ts=4: */
Rob Landleyaa872762005-10-28 13:05:12 +00002/*
Tim Rikerc1ef7bd2006-01-25 00:08:53 +00003 * tiny fuser implementation
4 *
Rob Landleyaa872762005-10-28 13:05:12 +00005 * Copyright 2004 Tony J. White
6 *
Denys Vlasenko0ef64bd2010-08-16 20:14:46 +02007 * Licensed under GPLv2, see file LICENSE in this source tree.
Rob Landleyaa872762005-10-28 13:05:12 +00008 */
9
Pere Orga5bc8c002011-04-11 03:29:49 +020010//usage:#define fuser_trivial_usage
11//usage: "[OPTIONS] FILE or PORT/PROTO"
12//usage:#define fuser_full_usage "\n\n"
13//usage: "Find processes which use FILEs or PORTs\n"
14//usage: "\nOptions:"
15//usage: "\n -m Find processes which use same fs as FILEs"
16//usage: "\n -4,-6 Search only IPv4/IPv6 space"
17//usage: "\n -s Don't display PIDs"
18//usage: "\n -k Kill found processes"
19//usage: "\n -SIGNAL Signal to send (default: KILL)"
20
Denis Vlasenkob6adbf12007-05-26 19:00:18 +000021#include "libbb.h"
Rob Landleyaa872762005-10-28 13:05:12 +000022
Denis Vlasenkoafc41132008-03-17 08:44:58 +000023#define MAX_LINE 255
Rob Landleyaa872762005-10-28 13:05:12 +000024
Denis Vlasenkoafc41132008-03-17 08:44:58 +000025#define OPTION_STRING "mks64"
26enum {
27 OPT_MOUNT = (1 << 0),
28 OPT_KILL = (1 << 1),
29 OPT_SILENT = (1 << 2),
30 OPT_IP6 = (1 << 3),
31 OPT_IP4 = (1 << 4),
32};
Rob Landleyaa872762005-10-28 13:05:12 +000033
34typedef struct inode_list {
Denis Vlasenkoafc41132008-03-17 08:44:58 +000035 struct inode_list *next;
Rob Landleyaa872762005-10-28 13:05:12 +000036 ino_t inode;
37 dev_t dev;
Rob Landleyaa872762005-10-28 13:05:12 +000038} inode_list;
39
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020040struct globals {
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +020041 int recursion_depth;
42 pid_t mypid;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020043 inode_list *inode_list_head;
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +020044 smallint kill_failed;
45 int killsig;
Denys Vlasenkoe8d0a142011-01-16 11:21:15 +010046} FIX_ALIASING;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020047#define G (*(struct globals*)&bb_common_bufsiz1)
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +020048#define INIT_G() do { \
49 G.mypid = getpid(); \
50 G.killsig = SIGKILL; \
51} while (0)
Rob Landleyaa872762005-10-28 13:05:12 +000052
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020053static void add_inode(const struct stat *st)
Rob Landleyaa872762005-10-28 13:05:12 +000054{
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020055 inode_list **curr = &G.inode_list_head;
56
57 while (*curr) {
58 if ((*curr)->dev == st->st_dev
59 && (*curr)->inode == st->st_ino
60 ) {
61 return;
62 }
63 curr = &(*curr)->next;
Tim Rikerc1ef7bd2006-01-25 00:08:53 +000064 }
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020065
66 *curr = xzalloc(sizeof(inode_list));
67 (*curr)->dev = st->st_dev;
68 (*curr)->inode = st->st_ino;
Rob Landleyaa872762005-10-28 13:05:12 +000069}
70
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +020071static smallint search_dev_inode(const struct stat *st)
Rob Landleyaa872762005-10-28 13:05:12 +000072{
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020073 inode_list *ilist = G.inode_list_head;
74
Denis Vlasenkoafc41132008-03-17 08:44:58 +000075 while (ilist) {
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020076 if (ilist->dev == st->st_dev) {
Denis Vlasenkoafc41132008-03-17 08:44:58 +000077 if (option_mask32 & OPT_MOUNT)
78 return 1;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020079 if (ilist->inode == st->st_ino)
Denis Vlasenkoafc41132008-03-17 08:44:58 +000080 return 1;
81 }
82 ilist = ilist->next;
Tim Rikerc1ef7bd2006-01-25 00:08:53 +000083 }
84 return 0;
Rob Landleyaa872762005-10-28 13:05:12 +000085}
86
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +020087enum {
88 PROC_NET = 0,
89 PROC_DIR,
90 PROC_DIR_LINKS,
91 PROC_SUBDIR_LINKS,
92};
93
94static smallint scan_proc_net_or_maps(const char *path, unsigned port)
Rob Landleyaa872762005-10-28 13:05:12 +000095{
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +020096 FILE *f;
97 char line[MAX_LINE + 1], addr[68];
98 int major, minor, r;
Eric Andersena68ea1c2006-01-30 22:48:39 +000099 long long uint64_inode;
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200100 unsigned tmp_port;
101 smallint retval;
102 struct stat statbuf;
103 const char *fmt;
104 void *fag, *sag;
Rob Landleyaa872762005-10-28 13:05:12 +0000105
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200106 f = fopen_for_read(path);
107 if (!f)
108 return 0;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200109
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200110 if (G.recursion_depth == PROC_NET) {
111 int fd;
112
113 /* find socket dev */
114 statbuf.st_dev = 0;
115 fd = socket(AF_INET, SOCK_DGRAM, 0);
116 if (fd >= 0) {
117 fstat(fd, &statbuf);
118 close(fd);
119 }
120
121 fmt = "%*d: %64[0-9A-Fa-f]:%x %*x:%*x %*x "
122 "%*x:%*x %*x:%*x %*x %*d %*d %llu";
123 fag = addr;
124 sag = &tmp_port;
125 } else {
126 fmt = "%*s %*s %*s %x:%x %llu";
127 fag = &major;
128 sag = &minor;
Rob Landleyaa872762005-10-28 13:05:12 +0000129 }
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200130
131 retval = 0;
132 while (fgets(line, MAX_LINE, f)) {
133 r = sscanf(line, fmt, fag, sag, &uint64_inode);
134 if (r != 3)
135 continue;
136
137 statbuf.st_ino = uint64_inode;
138 if (G.recursion_depth == PROC_NET) {
139 r = strlen(addr);
140 if (r == 8 && (option_mask32 & OPT_IP6))
141 continue;
142 if (r > 8 && (option_mask32 & OPT_IP4))
143 continue;
144 if (tmp_port == port)
145 add_inode(&statbuf);
146 } else {
147 if (major != 0 && minor != 0 && statbuf.st_ino != 0) {
148 statbuf.st_dev = makedev(major, minor);
149 retval = search_dev_inode(&statbuf);
150 if (retval)
151 break;
152 }
153 }
154 }
155 fclose(f);
156
157 return retval;
Rob Landleyaa872762005-10-28 13:05:12 +0000158}
159
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200160static smallint scan_recursive(const char *path)
Rob Landleyaa872762005-10-28 13:05:12 +0000161{
162 DIR *d;
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200163 struct dirent *d_ent;
164 smallint stop_scan;
165 smallint retval;
Rob Landleyaa872762005-10-28 13:05:12 +0000166
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200167 d = opendir(path);
168 if (d == NULL)
169 return 0;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200170
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200171 G.recursion_depth++;
172 retval = 0;
173 stop_scan = 0;
174 while (!stop_scan && (d_ent = readdir(d)) != NULL) {
175 struct stat statbuf;
176 pid_t pid;
177 char *subpath;
178
179 subpath = concat_subpath_file(path, d_ent->d_name);
180 if (subpath == NULL)
181 continue; /* . or .. */
182
183 switch (G.recursion_depth) {
184 case PROC_DIR:
185 pid = (pid_t)bb_strtou(d_ent->d_name, NULL, 10);
186 if (errno != 0
187 || pid == G.mypid
188 /* "this PID doesn't use specified FILEs or PORT/PROTO": */
189 || scan_recursive(subpath) == 0
190 ) {
191 break;
192 }
193 if (option_mask32 & OPT_KILL) {
194 if (kill(pid, G.killsig) != 0) {
195 bb_perror_msg("kill pid %s", d_ent->d_name);
196 G.kill_failed = 1;
197 }
198 }
199 if (!(option_mask32 & OPT_SILENT))
200 printf("%s ", d_ent->d_name);
201 retval = 1;
202 break;
203
204 case PROC_DIR_LINKS:
205 switch (
206 index_in_substrings(
207 "cwd" "\0" "exe" "\0"
208 "root" "\0" "fd" "\0"
209 "lib" "\0" "mmap" "\0"
210 "maps" "\0",
211 d_ent->d_name
212 )
213 ) {
214 enum {
215 CWD_LINK,
216 EXE_LINK,
217 ROOT_LINK,
218 FD_DIR_LINKS,
219 LIB_DIR_LINKS,
220 MMAP_DIR_LINKS,
221 MAPS,
222 };
223 case CWD_LINK:
224 case EXE_LINK:
225 case ROOT_LINK:
226 goto scan_link;
227 case FD_DIR_LINKS:
228 case LIB_DIR_LINKS:
229 case MMAP_DIR_LINKS:
230 stop_scan = scan_recursive(subpath);
231 if (stop_scan)
232 retval = stop_scan;
233 break;
234 case MAPS:
235 stop_scan = scan_proc_net_or_maps(subpath, 0);
236 if (stop_scan)
237 retval = stop_scan;
238 default:
239 break;
240 }
241 break;
242 case PROC_SUBDIR_LINKS:
243 scan_link:
244 if (stat(subpath, &statbuf) < 0)
245 break;
246 stop_scan = search_dev_inode(&statbuf);
247 if (stop_scan)
248 retval = stop_scan;
249 default:
250 break;
251 }
252 free(subpath);
Rob Landleyaa872762005-10-28 13:05:12 +0000253 }
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000254 closedir(d);
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200255 G.recursion_depth--;
256 return retval;
Rob Landleyaa872762005-10-28 13:05:12 +0000257}
258
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +0000259int fuser_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
Denis Vlasenkoa60f84e2008-07-05 09:18:54 +0000260int fuser_main(int argc UNUSED_PARAM, char **argv)
Rob Landley31642d72006-03-14 21:45:38 +0000261{
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000262 char **pp;
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200263
264 INIT_G();
265
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000266 /* Handle -SIGNAL. Oh my... */
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000267 pp = argv;
268 while (*++pp) {
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200269 int sig;
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000270 char *arg = *pp;
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200271
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000272 if (arg[0] != '-')
273 continue;
274 if (arg[1] == '-' && arg[2] == '\0') /* "--" */
275 break;
276 if ((arg[1] == '4' || arg[1] == '6') && arg[2] == '\0')
277 continue; /* it's "-4" or "-6" */
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200278 sig = get_signum(&arg[1]);
279 if (sig < 0)
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000280 continue;
281 /* "-SIGNAL" option found. Remove it and bail out */
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200282 G.killsig = sig;
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000283 do {
284 pp[0] = arg = pp[1];
285 pp++;
286 } while (arg);
287 break;
288 }
Rob Landleyaa872762005-10-28 13:05:12 +0000289
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200290 opt_complementary = "-1"; /* at least one param */
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200291 getopt32(argv, OPTION_STRING);
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000292 argv += optind;
Mike Frysinger70cbb6e2006-04-21 22:04:05 +0000293
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000294 pp = argv;
295 while (*pp) {
Maksym Kryzhanovskyye3657dc2010-06-06 22:56:12 +0200296 /* parse net arg */
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200297 unsigned port;
298 char path[sizeof("/proc/net/TCP6")];
299
300 strcpy(path, "/proc/net/");
301 if (sscanf(*pp, "%u/%4s", &port, path + sizeof("/proc/net/")-1) == 2
Denys Vlasenko217a7f42011-05-29 02:03:38 +0200302 && access(path, R_OK) == 0
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200303 ) {
304 /* PORT/PROTO */
305 scan_proc_net_or_maps(path, port);
306 } else {
307 /* FILE */
308 struct stat statbuf;
309 xstat(*pp, &statbuf);
310 add_inode(&statbuf);
Rob Landleyaa872762005-10-28 13:05:12 +0000311 }
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000312 pp++;
Rob Landleyaa872762005-10-28 13:05:12 +0000313 }
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000314
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200315 if (scan_recursive("/proc")) {
316 if (!(option_mask32 & OPT_SILENT))
317 bb_putchar('\n');
318 return G.kill_failed;
Rob Landleyaa872762005-10-28 13:05:12 +0000319 }
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200320
Maksym Kryzhanovskyyeeed2302011-05-23 03:39:48 +0200321 return EXIT_FAILURE;
Tim Rikerc1ef7bd2006-01-25 00:08:53 +0000322}