blob: a1b93d77fe9fcadbff0097467000248f14822b09 [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
Denis Vlasenkob6adbf12007-05-26 19:00:18 +000010#include "libbb.h"
Rob Landleyaa872762005-10-28 13:05:12 +000011
Denis Vlasenkoafc41132008-03-17 08:44:58 +000012#define MAX_LINE 255
Rob Landleyaa872762005-10-28 13:05:12 +000013
Denis Vlasenkoafc41132008-03-17 08:44:58 +000014#define OPTION_STRING "mks64"
15enum {
16 OPT_MOUNT = (1 << 0),
17 OPT_KILL = (1 << 1),
18 OPT_SILENT = (1 << 2),
19 OPT_IP6 = (1 << 3),
20 OPT_IP4 = (1 << 4),
21};
Rob Landleyaa872762005-10-28 13:05:12 +000022
23typedef struct inode_list {
Denis Vlasenkoafc41132008-03-17 08:44:58 +000024 struct inode_list *next;
Rob Landleyaa872762005-10-28 13:05:12 +000025 ino_t inode;
26 dev_t dev;
Rob Landleyaa872762005-10-28 13:05:12 +000027} inode_list;
28
29typedef struct pid_list {
Rob Landleyaa872762005-10-28 13:05:12 +000030 struct pid_list *next;
Denis Vlasenkoafc41132008-03-17 08:44:58 +000031 pid_t pid;
Rob Landleyaa872762005-10-28 13:05:12 +000032} pid_list;
33
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020034
35struct globals {
36 pid_list *pid_list_head;
37 inode_list *inode_list_head;
Denys Vlasenkoe8d0a142011-01-16 11:21:15 +010038} FIX_ALIASING;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020039#define G (*(struct globals*)&bb_common_bufsiz1)
40#define INIT_G() do { } while (0)
41
42
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020043static void add_pid(const pid_t pid)
Rob Landleyaa872762005-10-28 13:05:12 +000044{
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020045 pid_list **curr = &G.pid_list_head;
46
47 while (*curr) {
48 if ((*curr)->pid == pid)
49 return;
50 curr = &(*curr)->next;
Tim Rikerc1ef7bd2006-01-25 00:08:53 +000051 }
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020052
53 *curr = xzalloc(sizeof(pid_list));
54 (*curr)->pid = pid;
Rob Landleyaa872762005-10-28 13:05:12 +000055}
56
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020057static void add_inode(const struct stat *st)
Rob Landleyaa872762005-10-28 13:05:12 +000058{
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020059 inode_list **curr = &G.inode_list_head;
60
61 while (*curr) {
62 if ((*curr)->dev == st->st_dev
63 && (*curr)->inode == st->st_ino
64 ) {
65 return;
66 }
67 curr = &(*curr)->next;
Tim Rikerc1ef7bd2006-01-25 00:08:53 +000068 }
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020069
70 *curr = xzalloc(sizeof(inode_list));
71 (*curr)->dev = st->st_dev;
72 (*curr)->inode = st->st_ino;
Rob Landleyaa872762005-10-28 13:05:12 +000073}
74
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020075static void scan_proc_net(const char *path, unsigned port)
Rob Landleyaa872762005-10-28 13:05:12 +000076{
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020077 char line[MAX_LINE + 1];
Denis Vlasenko50f7f442007-04-11 23:20:53 +000078 long long uint64_inode;
Denis Vlasenkoafc41132008-03-17 08:44:58 +000079 unsigned tmp_port;
Rob Landleyaa872762005-10-28 13:05:12 +000080 FILE *f;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020081 struct stat st;
Maksym Kryzhanovskyye3657dc2010-06-06 22:56:12 +020082 int fd;
Rob Landleyaa872762005-10-28 13:05:12 +000083
Maksym Kryzhanovskyye3657dc2010-06-06 22:56:12 +020084 /* find socket dev */
85 st.st_dev = 0;
86 fd = socket(AF_INET, SOCK_DGRAM, 0);
87 if (fd >= 0) {
88 fstat(fd, &st);
89 close(fd);
90 }
Rob Landleyaa872762005-10-28 13:05:12 +000091
Denis Vlasenko5415c852008-07-21 23:05:26 +000092 f = fopen_for_read(path);
Denis Vlasenko50f7f442007-04-11 23:20:53 +000093 if (!f)
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +020094 return;
Denis Vlasenkoafc41132008-03-17 08:44:58 +000095
96 while (fgets(line, MAX_LINE, f)) {
Denis Vlasenko6b797182008-07-12 09:32:38 +000097 char addr[68];
Denis Vlasenko50f7f442007-04-11 23:20:53 +000098 if (sscanf(line, "%*d: %64[0-9A-Fa-f]:%x %*x:%*x %*x %*x:%*x "
99 "%*x:%*x %*x %*d %*d %llu",
100 addr, &tmp_port, &uint64_inode) == 3
101 ) {
Denis Vlasenkoa46dd892008-07-12 09:20:44 +0000102 int len = strlen(addr);
103 if (len == 8 && (option_mask32 & OPT_IP6))
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000104 continue;
Denis Vlasenkoa46dd892008-07-12 09:20:44 +0000105 if (len > 8 && (option_mask32 & OPT_IP4))
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000106 continue;
107 if (tmp_port == port) {
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200108 st.st_ino = uint64_inode;
109 add_inode(&st);
Rob Landleyaa872762005-10-28 13:05:12 +0000110 }
111 }
Rob Landleyaa872762005-10-28 13:05:12 +0000112 }
113 fclose(f);
Rob Landleyaa872762005-10-28 13:05:12 +0000114}
115
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200116static int search_dev_inode(const struct stat *st)
Rob Landleyaa872762005-10-28 13:05:12 +0000117{
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200118 inode_list *ilist = G.inode_list_head;
119
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000120 while (ilist) {
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200121 if (ilist->dev == st->st_dev) {
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000122 if (option_mask32 & OPT_MOUNT)
123 return 1;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200124 if (ilist->inode == st->st_ino)
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000125 return 1;
126 }
127 ilist = ilist->next;
Tim Rikerc1ef7bd2006-01-25 00:08:53 +0000128 }
129 return 0;
Rob Landleyaa872762005-10-28 13:05:12 +0000130}
131
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200132static void scan_pid_maps(const char *fname, pid_t pid)
Rob Landleyaa872762005-10-28 13:05:12 +0000133{
134 FILE *file;
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000135 char line[MAX_LINE + 1];
Rob Landleyaa872762005-10-28 13:05:12 +0000136 int major, minor;
Eric Andersena68ea1c2006-01-30 22:48:39 +0000137 long long uint64_inode;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200138 struct stat st;
Rob Landleyaa872762005-10-28 13:05:12 +0000139
Denis Vlasenko5415c852008-07-21 23:05:26 +0000140 file = fopen_for_read(fname);
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000141 if (!file)
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200142 return;
143
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000144 while (fgets(line, MAX_LINE, file)) {
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000145 if (sscanf(line, "%*s %*s %*s %x:%x %llu", &major, &minor, &uint64_inode) != 3)
146 continue;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200147 st.st_ino = uint64_inode;
148 if (major == 0 && minor == 0 && st.st_ino == 0)
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000149 continue;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200150 st.st_dev = makedev(major, minor);
151 if (search_dev_inode(&st))
152 add_pid(pid);
Rob Landleyaa872762005-10-28 13:05:12 +0000153 }
154 fclose(file);
Rob Landleyaa872762005-10-28 13:05:12 +0000155}
156
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200157static void scan_link(const char *lname, pid_t pid)
Rob Landleyaa872762005-10-28 13:05:12 +0000158{
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200159 struct stat st;
Rob Landleyaa872762005-10-28 13:05:12 +0000160
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200161 if (stat(lname, &st) >= 0) {
162 if (search_dev_inode(&st))
163 add_pid(pid);
164 }
Rob Landleyaa872762005-10-28 13:05:12 +0000165}
166
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200167static void scan_dir_links(const char *dname, pid_t pid)
Rob Landleyaa872762005-10-28 13:05:12 +0000168{
169 DIR *d;
170 struct dirent *de;
171 char *lname;
172
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000173 d = opendir(dname);
174 if (!d)
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200175 return;
176
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000177 while ((de = readdir(d)) != NULL) {
178 lname = concat_subpath_file(dname, de->d_name);
179 if (lname == NULL)
180 continue;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200181 scan_link(lname, pid);
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000182 free(lname);
Rob Landleyaa872762005-10-28 13:05:12 +0000183 }
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000184 closedir(d);
Tim Rikerc1ef7bd2006-01-25 00:08:53 +0000185}
Rob Landleyaa872762005-10-28 13:05:12 +0000186
Denis Vlasenko5de8a132008-05-28 12:44:22 +0000187/* NB: does chdir internally */
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200188static void scan_proc_pids(void)
Rob Landleyaa872762005-10-28 13:05:12 +0000189{
190 DIR *d;
191 struct dirent *de;
192 pid_t pid;
Rob Landleyaa872762005-10-28 13:05:12 +0000193
Denis Vlasenkocd9d4c82008-05-28 14:57:58 +0000194 xchdir("/proc");
Denis Vlasenko5de8a132008-05-28 12:44:22 +0000195 d = opendir("/proc");
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000196 if (!d)
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200197 return;
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000198
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000199 while ((de = readdir(d)) != NULL) {
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000200 pid = (pid_t)bb_strtou(de->d_name, NULL, 10);
201 if (errno)
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000202 continue;
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000203 if (chdir(de->d_name) < 0)
Tim Rikerc1ef7bd2006-01-25 00:08:53 +0000204 continue;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200205 scan_link("cwd", pid);
206 scan_link("exe", pid);
207 scan_link("root", pid);
208
209 scan_dir_links("fd", pid);
210 scan_dir_links("lib", pid);
211 scan_dir_links("mmap", pid);
212
213 scan_pid_maps("maps", pid);
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000214 xchdir("/proc");
Tim Rikerc1ef7bd2006-01-25 00:08:53 +0000215 }
216 closedir(d);
Rob Landleyaa872762005-10-28 13:05:12 +0000217}
218
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +0000219int fuser_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
Denis Vlasenkoa60f84e2008-07-05 09:18:54 +0000220int fuser_main(int argc UNUSED_PARAM, char **argv)
Rob Landley31642d72006-03-14 21:45:38 +0000221{
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000222 pid_list *plist;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200223 pid_t mypid;
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000224 char **pp;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200225 struct stat st;
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000226 unsigned port;
227 int opt;
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200228 int exitcode;
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000229 int killsig;
230/*
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200231fuser [OPTIONS] FILE or PORT/PROTO
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000232Find processes which use FILEs or PORTs
233 -m Find processes which use same fs as FILEs
234 -4 Search only IPv4 space
235 -6 Search only IPv6 space
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200236 -s Don't display PIDs
237 -k Kill found processes
238 -SIGNAL Signal to send (default: KILL)
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000239*/
240 /* Handle -SIGNAL. Oh my... */
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200241 killsig = SIGKILL; /* yes, the default is not SIGTERM */
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000242 pp = argv;
243 while (*++pp) {
244 char *arg = *pp;
245 if (arg[0] != '-')
246 continue;
247 if (arg[1] == '-' && arg[2] == '\0') /* "--" */
248 break;
249 if ((arg[1] == '4' || arg[1] == '6') && arg[2] == '\0')
250 continue; /* it's "-4" or "-6" */
251 opt = get_signum(&arg[1]);
252 if (opt < 0)
253 continue;
254 /* "-SIGNAL" option found. Remove it and bail out */
255 killsig = opt;
256 do {
257 pp[0] = arg = pp[1];
258 pp++;
259 } while (arg);
260 break;
261 }
Rob Landleyaa872762005-10-28 13:05:12 +0000262
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200263 opt_complementary = "-1"; /* at least one param */
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000264 opt = getopt32(argv, OPTION_STRING);
265 argv += optind;
Mike Frysinger70cbb6e2006-04-21 22:04:05 +0000266
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000267 pp = argv;
268 while (*pp) {
Maksym Kryzhanovskyye3657dc2010-06-06 22:56:12 +0200269 /* parse net arg */
270 char path[20], tproto[5];
271 if (sscanf(*pp, "%u/%4s", &port, tproto) != 2)
272 goto file;
273 sprintf(path, "/proc/net/%s", tproto);
274 if (access(path, R_OK) != 0) { /* PORT/PROTO */
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200275 scan_proc_net(path, port);
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000276 } else { /* FILE */
Maksym Kryzhanovskyye3657dc2010-06-06 22:56:12 +0200277 file:
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200278 xstat(*pp, &st);
279 add_inode(&st);
Rob Landleyaa872762005-10-28 13:05:12 +0000280 }
Denis Vlasenkoafc41132008-03-17 08:44:58 +0000281 pp++;
Rob Landleyaa872762005-10-28 13:05:12 +0000282 }
Denis Vlasenko50f7f442007-04-11 23:20:53 +0000283
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200284 scan_proc_pids(); /* changes dir to "/proc" */
Rob Landleyaa872762005-10-28 13:05:12 +0000285
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200286 mypid = getpid();
287 plist = G.pid_list_head;
288 while (1) {
289 if (!plist)
290 return EXIT_FAILURE;
291 if (plist->pid != mypid)
292 break;
293 plist = plist->next;
Rob Landleyaa872762005-10-28 13:05:12 +0000294 }
Maksym Kryzhanovskyyfef9ee72010-05-22 20:41:08 +0200295
296 exitcode = EXIT_SUCCESS;
297 do {
298 if (plist->pid != mypid) {
299 if (opt & OPT_KILL) {
300 if (kill(plist->pid, killsig) != 0) {
301 bb_perror_msg("kill pid %u", (unsigned)plist->pid);
302 exitcode = EXIT_FAILURE;
303 }
304 }
305 if (!(opt & OPT_SILENT)) {
306 printf("%u ", (unsigned)plist->pid);
307 }
308 }
309 plist = plist->next;
310 } while (plist);
311
312 if (!(opt & (OPT_SILENT))) {
313 bb_putchar('\n');
314 }
315
316 return exitcode;
Tim Rikerc1ef7bd2006-01-25 00:08:53 +0000317}