blob: 7041f7c59c2e6f6c80a44d02c1b44d63eff2a96d [file] [log] [blame]
Rob Landleyd00b3a52005-08-20 05:07:08 +00001/* vi: set sw=4 ts=4: */
2/*
3 * mountpoint implementation for busybox
4 *
Bernhard Reutner-Fischer6c4dade2008-09-25 12:13:34 +00005 * Copyright (C) 2005 Bernhard Reutner-Fischer
Rob Landleyd00b3a52005-08-20 05:07:08 +00006 *
Denys Vlasenko0ef64bd2010-08-16 20:14:46 +02007 * Licensed under GPLv2 or later, see file LICENSE in this source tree.
Rob Landleyd00b3a52005-08-20 05:07:08 +00008 *
9 * Based on sysvinit's mountpoint
10 */
11
Pere Orga5bc8c002011-04-11 03:29:49 +020012//usage:#define mountpoint_trivial_usage
13//usage: "[-q] <[-dn] DIR | -x DEVICE>"
14//usage:#define mountpoint_full_usage "\n\n"
15//usage: "Check if the directory is a mountpoint\n"
Pere Orga5bc8c002011-04-11 03:29:49 +020016//usage: "\n -q Quiet"
17//usage: "\n -d Print major/minor device number of the filesystem"
18//usage: "\n -n Print device name of the filesystem"
19//usage: "\n -x Print major/minor device number of the blockdevice"
20//usage:
21//usage:#define mountpoint_example_usage
22//usage: "$ mountpoint /proc\n"
23//usage: "/proc is not a mountpoint\n"
24//usage: "$ mountpoint /sys\n"
25//usage: "/sys is a mountpoint\n"
26
Denis Vlasenkob6adbf12007-05-26 19:00:18 +000027#include "libbb.h"
Rob Landleyd00b3a52005-08-20 05:07:08 +000028
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +000029int mountpoint_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
Denis Vlasenko7a1ddf22008-11-29 12:54:16 +000030int mountpoint_main(int argc UNUSED_PARAM, char **argv)
Rob Landleyd00b3a52005-08-20 05:07:08 +000031{
Denis Vlasenkof0ed3762006-10-26 23:21:47 +000032 struct stat st;
Denis Vlasenko7a1ddf22008-11-29 12:54:16 +000033 const char *msg;
Denis Vlasenkof0ed3762006-10-26 23:21:47 +000034 char *arg;
Denis Vlasenko7a1ddf22008-11-29 12:54:16 +000035 int rc, opt;
36
37 opt_complementary = "=1"; /* must have one argument */
38 opt = getopt32(argv, "qdxn");
Rob Landleyd00b3a52005-08-20 05:07:08 +000039#define OPT_q (1)
40#define OPT_d (2)
41#define OPT_x (4)
Denis Vlasenko7a1ddf22008-11-29 12:54:16 +000042#define OPT_n (8)
Denis Vlasenkof0ed3762006-10-26 23:21:47 +000043 arg = argv[optind];
Denis Vlasenko7a1ddf22008-11-29 12:54:16 +000044 msg = "%s";
Tim Rikerc1ef7bd2006-01-25 00:08:53 +000045
Denis Vlasenko7a1ddf22008-11-29 12:54:16 +000046 rc = (opt & OPT_x) ? stat(arg, &st) : lstat(arg, &st);
47 if (rc != 0)
48 goto err;
Denis Vlasenkof0ed3762006-10-26 23:21:47 +000049
Denis Vlasenko7a1ddf22008-11-29 12:54:16 +000050 if (opt & OPT_x) {
51 if (S_ISBLK(st.st_mode)) {
52 printf("%u:%u\n", major(st.st_rdev),
53 minor(st.st_rdev));
54 return EXIT_SUCCESS;
Rob Landleyd00b3a52005-08-20 05:07:08 +000055 }
Denis Vlasenko7a1ddf22008-11-29 12:54:16 +000056 errno = 0; /* make perror_msg work as error_msg */
57 msg = "%s: not a block device";
58 goto err;
Rob Landleyd00b3a52005-08-20 05:07:08 +000059 }
Denis Vlasenko7a1ddf22008-11-29 12:54:16 +000060
61 errno = ENOTDIR;
62 if (S_ISDIR(st.st_mode)) {
63 dev_t st_dev = st.st_dev;
64 ino_t st_ino = st.st_ino;
65 char *p = xasprintf("%s/..", arg);
66
67 if (stat(p, &st) == 0) {
68 //int is_mnt = (st_dev != st.st_dev) || (st_dev == st.st_dev && st_ino == st.st_ino);
69 int is_not_mnt = (st_dev == st.st_dev) && (st_ino != st.st_ino);
70
71 if (opt & OPT_d)
72 printf("%u:%u\n", major(st_dev), minor(st_dev));
Vladimir Dronnikov4214f8b2009-11-01 04:33:23 +010073 if (opt & OPT_n) {
74 const char *d = find_block_device(arg);
75 /* name is undefined, but device is mounted -> anonymous superblock! */
76 /* happens with btrfs */
77 if (!d) {
78 d = "UNKNOWN";
79 /* TODO: iterate /proc/mounts, or /proc/self/mountinfo
80 * to find out the device name */
81 }
82 printf("%s %s\n", d, arg);
83 }
Denis Vlasenko7a1ddf22008-11-29 12:54:16 +000084 if (!(opt & (OPT_q | OPT_d | OPT_n)))
85 printf("%s is %sa mountpoint\n", arg, is_not_mnt ? "not " : "");
86 return is_not_mnt;
87 }
88 arg = p;
89 /* else: stat had set errno, just fall through */
90 }
91
92 err:
Denis Vlasenkof0ed3762006-10-26 23:21:47 +000093 if (!(opt & OPT_q))
Denis Vlasenko7a1ddf22008-11-29 12:54:16 +000094 bb_perror_msg(msg, arg);
Denis Vlasenkof0ed3762006-10-26 23:21:47 +000095 return EXIT_FAILURE;
Rob Landleyd00b3a52005-08-20 05:07:08 +000096}