blob: ee6fe15e63495b78672a5ae3d6d41909a931ffe9 [file] [log] [blame]
Eric Andersencc8ed391999-10-05 16:24:54 +00001/*
Eric Andersen596e5461999-10-07 08:30:23 +00002 * Mini mount implementation for busybox
3 *
Eric Andersenc4996011999-10-20 22:08:37 +00004 * Copyright (C) 1995, 1996 by Bruce Perens <bruce@pixar.com>.
Eric Andersen596e5461999-10-07 08:30:23 +00005 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19 *
20 * 3/21/1999 Charles P. Wright <cpwright@cpwright.com>
21 * searches through fstab when -a is passed
22 * will try mounting stuff with all fses when passed -t auto
23 *
24 * 1999-04-17 Dave Cinege...Rewrote -t auto. Fixed ro mtab.
Eric Andersenc4996011999-10-20 22:08:37 +000025 *
26 * 1999-10-07 Erik Andersen <andersen@lineo.com>, <andersee@debian.org>.
27 * Rewrote of a lot of code. Removed mtab usage (I plan on
28 * putting it back as a compile-time option some time),
29 * major adjustments to option parsing, and some serious
30 * dieting all around.
Eric Andersencc8ed391999-10-05 16:24:54 +000031*/
32
33#include "internal.h"
34#include <stdlib.h>
35#include <unistd.h>
36#include <errno.h>
37#include <string.h>
38#include <stdio.h>
39#include <mntent.h>
40#include <sys/mount.h>
41#include <ctype.h>
Eric Andersen596e5461999-10-07 08:30:23 +000042#include <fstab.h>
Eric Andersencc8ed391999-10-05 16:24:54 +000043
Eric Andersend0246fb1999-11-04 21:18:07 +000044extern const char mtab_file[]; /* Defined in utility.c */
45
Eric Andersend73dc5b1999-11-10 23:13:02 +000046static const char mount_usage[] = "\tmount [flags]\n"
Eric Andersen596e5461999-10-07 08:30:23 +000047 "\tmount [flags] device directory [-o options,more-options]\n"
48 "\n"
49 "Flags:\n"
50 "\t-a:\tMount all file systems in fstab.\n"
Eric Andersend0246fb1999-11-04 21:18:07 +000051#ifdef BB_MTAB
52 "\t-f:\t\"Fake\" mount. Add entry to mount table but don't mount it.\n"
53 "\t-n:\tDon't write a mount table entry.\n"
54#endif
Eric Andersen596e5461999-10-07 08:30:23 +000055 "\t-o option:\tOne of many filesystem options, listed below.\n"
56 "\t-r:\tMount the filesystem read-only.\n"
57 "\t-t filesystem-type:\tSpecify the filesystem type.\n"
58 "\t-w:\tMount for reading and writing (default).\n"
59 "\n"
60 "Options for use with the \"-o\" flag:\n"
61 "\tasync / sync:\tWrites are asynchronous / synchronous.\n"
62 "\tdev / nodev:\tAllow use of special device files / disallow them.\n"
63 "\texec / noexec:\tAllow use of executable files / disallow them.\n"
64 "\tsuid / nosuid:\tAllow set-user-id-root programs / disallow them.\n"
65 "\tremount: Re-mount a currently-mounted filesystem, changing its flags.\n"
66 "\tro / rw: Mount for read-only / read-write.\n"
67 "\t"
68 "There are EVEN MORE flags that are specific to each filesystem.\n"
69 "You'll have to see the written documentation for those.\n";
Eric Andersencc8ed391999-10-05 16:24:54 +000070
Eric Andersend0246fb1999-11-04 21:18:07 +000071
Eric Andersencc8ed391999-10-05 16:24:54 +000072struct mount_options {
Eric Andersen596e5461999-10-07 08:30:23 +000073 const char *name;
74 unsigned long and;
75 unsigned long or;
Eric Andersencc8ed391999-10-05 16:24:54 +000076};
77
Eric Andersen596e5461999-10-07 08:30:23 +000078static const struct mount_options mount_options[] = {
79 {"async", ~MS_SYNCHRONOUS, 0},
80 {"defaults", ~0, 0},
81 {"dev", ~MS_NODEV, 0},
82 {"exec", ~MS_NOEXEC, 0},
83 {"nodev", ~0, MS_NODEV},
84 {"noexec", ~0, MS_NOEXEC},
85 {"nosuid", ~0, MS_NOSUID},
86 {"remount", ~0, MS_REMOUNT},
87 {"ro", ~0, MS_RDONLY},
88 {"rw", ~MS_RDONLY, 0},
89 {"suid", ~MS_NOSUID, 0},
90 {"sync", ~0, MS_SYNCHRONOUS},
91 {0, 0, 0}
Eric Andersencc8ed391999-10-05 16:24:54 +000092};
93
Eric Andersend0246fb1999-11-04 21:18:07 +000094#if ! defined BB_MTAB
Eric Andersena9c95ea1999-11-15 17:33:30 +000095#define do_mount(specialfile, dir, filesystemtype, flags, string_flags, useMtab, fakeIt, mtab_opts) \
Eric Andersend0246fb1999-11-04 21:18:07 +000096 mount(specialfile, dir, filesystemtype, flags, string_flags)
97#else
98static int
99do_mount(char* specialfile, char* dir, char* filesystemtype,
Eric Andersena9c95ea1999-11-15 17:33:30 +0000100 long flags, void* string_flags, int useMtab, int fakeIt, char* mtab_opts)
Eric Andersend0246fb1999-11-04 21:18:07 +0000101{
102 int status=0;
103
104 if (fakeIt==FALSE)
105 status=mount(specialfile, dir, filesystemtype, flags, string_flags);
106
107 if ( status == 0 ) {
108 if ( useMtab==TRUE )
Eric Andersena9c95ea1999-11-15 17:33:30 +0000109 write_mtab(specialfile, dir, filesystemtype, flags, mtab_opts);
Eric Andersend0246fb1999-11-04 21:18:07 +0000110 return 0;
111 }
112 else
113 return( status);
114}
115#endif
116
Eric Andersencc8ed391999-10-05 16:24:54 +0000117
Erik Andersen0881de72000-01-05 09:34:26 +0000118#if defined BB_MTAB
119#define whine_if_fstab_is_missing() {}
120#else
121extern void whine_if_fstab_is_missing()
122{
123 struct stat statBuf;
124 if (stat("/etc/fstab", &statBuf) < 0)
125 fprintf(stderr, "/etc/fstab file missing -- install one to name /dev/root.\n\n");
126}
127#endif
128
129
Eric Andersen8341a151999-10-08 17:14:14 +0000130/* Seperate standard mount options from the nonstandard string options */
Eric Andersencc8ed391999-10-05 16:24:54 +0000131static void
Eric Andersen8341a151999-10-08 17:14:14 +0000132parse_mount_options ( char *options, unsigned long *flags, char *strflags)
Eric Andersencc8ed391999-10-05 16:24:54 +0000133{
Eric Andersen8341a151999-10-08 17:14:14 +0000134 while (options) {
135 int gotone=FALSE;
Eric Andersen596e5461999-10-07 08:30:23 +0000136 char *comma = strchr (options, ',');
137 const struct mount_options* f = mount_options;
138 if (comma)
139 *comma = '\0';
Eric Andersencc8ed391999-10-05 16:24:54 +0000140
Eric Andersen596e5461999-10-07 08:30:23 +0000141 while (f->name != 0) {
Eric Andersen596e5461999-10-07 08:30:23 +0000142 if (strcasecmp (f->name, options) == 0) {
Eric Andersen3ae0c781999-11-04 01:13:21 +0000143
Eric Andersen596e5461999-10-07 08:30:23 +0000144 *flags &= f->and;
145 *flags |= f->or;
Eric Andersen8341a151999-10-08 17:14:14 +0000146 gotone=TRUE;
147 break;
Eric Andersen596e5461999-10-07 08:30:23 +0000148 }
149 f++;
Eric Andersencc8ed391999-10-05 16:24:54 +0000150 }
Eric Andersen8341a151999-10-08 17:14:14 +0000151 if (*strflags && strflags!= '\0' && gotone==FALSE) {
152 char *temp=strflags;
153 temp += strlen (strflags);
154 *temp++ = ',';
155 *temp++ = '\0';
Eric Andersencc8ed391999-10-05 16:24:54 +0000156 }
Eric Andersen8341a151999-10-08 17:14:14 +0000157 if (gotone==FALSE) {
158 strcat (strflags, options);
159 gotone=FALSE;
160 }
Eric Andersen596e5461999-10-07 08:30:23 +0000161 if (comma) {
162 *comma = ',';
163 options = ++comma;
Eric Andersen8341a151999-10-08 17:14:14 +0000164 } else {
Eric Andersen596e5461999-10-07 08:30:23 +0000165 break;
Eric Andersen8341a151999-10-08 17:14:14 +0000166 }
Eric Andersen596e5461999-10-07 08:30:23 +0000167 }
Eric Andersencc8ed391999-10-05 16:24:54 +0000168}
169
170int
Eric Andersend0246fb1999-11-04 21:18:07 +0000171mount_one(char *blockDevice, char *directory, char *filesystemType,
Eric Andersena9c95ea1999-11-15 17:33:30 +0000172 unsigned long flags, char *string_flags, int useMtab, int fakeIt, char *mtab_opts)
Eric Andersencc8ed391999-10-05 16:24:54 +0000173{
Eric Andersen596e5461999-10-07 08:30:23 +0000174 int status = 0;
Eric Andersencc8ed391999-10-05 16:24:54 +0000175
Eric Andersen596e5461999-10-07 08:30:23 +0000176 char buf[255];
Eric Andersencc8ed391999-10-05 16:24:54 +0000177
Eric Andersen0ecb54a1999-12-05 23:24:55 +0000178#if defined BB_FEATURE_USE_PROCFS
Eric Andersen596e5461999-10-07 08:30:23 +0000179 if (strcmp(filesystemType, "auto") == 0) {
180 FILE *f = fopen ("/proc/filesystems", "r");
Eric Andersencc8ed391999-10-05 16:24:54 +0000181
Eric Andersen596e5461999-10-07 08:30:23 +0000182 if (f == NULL)
183 return( FALSE);
Eric Andersencc8ed391999-10-05 16:24:54 +0000184
Eric Andersen596e5461999-10-07 08:30:23 +0000185 while (fgets (buf, sizeof (buf), f) != NULL) {
186 filesystemType = buf;
187 if (*filesystemType == '\t') { // Not a nodev filesystem
Eric Andersencc8ed391999-10-05 16:24:54 +0000188
Eric Andersen596e5461999-10-07 08:30:23 +0000189 // Add NULL termination to each line
190 while (*filesystemType && *filesystemType != '\n')
191 filesystemType++;
192 *filesystemType = '\0';
Eric Andersencc8ed391999-10-05 16:24:54 +0000193
Eric Andersen596e5461999-10-07 08:30:23 +0000194 filesystemType = buf;
195 filesystemType++; // hop past tab
196
Eric Andersend0246fb1999-11-04 21:18:07 +0000197 status = do_mount (blockDevice, directory, filesystemType,
Eric Andersena9c95ea1999-11-15 17:33:30 +0000198 flags | MS_MGC_VAL, string_flags, useMtab,
199 fakeIt, mtab_opts);
Eric Andersen596e5461999-10-07 08:30:23 +0000200 if (status == 0)
201 break;
202 }
Eric Andersencc8ed391999-10-05 16:24:54 +0000203 }
Eric Andersen596e5461999-10-07 08:30:23 +0000204 fclose (f);
Eric Andersen0ecb54a1999-12-05 23:24:55 +0000205 } else
206#endif
207 {
Eric Andersend0246fb1999-11-04 21:18:07 +0000208 status = do_mount (blockDevice, directory, filesystemType,
Eric Andersena9c95ea1999-11-15 17:33:30 +0000209 flags | MS_MGC_VAL, string_flags, useMtab,
210 fakeIt, mtab_opts);
Eric Andersen596e5461999-10-07 08:30:23 +0000211 }
Eric Andersencc8ed391999-10-05 16:24:54 +0000212
Eric Andersen596e5461999-10-07 08:30:23 +0000213 if (status) {
214 fprintf (stderr, "Mounting %s on %s failed: %s\n",
215 blockDevice, directory, strerror(errno));
216 return (FALSE);
217 }
218 return (TRUE);
Eric Andersencc8ed391999-10-05 16:24:54 +0000219}
220
Eric Andersen596e5461999-10-07 08:30:23 +0000221extern int mount_main (int argc, char **argv)
Eric Andersencc8ed391999-10-05 16:24:54 +0000222{
Eric Andersena9c95ea1999-11-15 17:33:30 +0000223 char string_flags_buf[1024]="";
224 char *string_flags = string_flags_buf;
225 char *extra_opts = string_flags_buf;
Eric Andersen596e5461999-10-07 08:30:23 +0000226 unsigned long flags = 0;
227 char *filesystemType = "auto";
Eric Andersen8341a151999-10-08 17:14:14 +0000228 char *device = NULL;
229 char *directory = NULL;
Eric Andersend0246fb1999-11-04 21:18:07 +0000230 int all = FALSE;
231 int fakeIt = FALSE;
232 int useMtab = TRUE;
Eric Andersen8341a151999-10-08 17:14:14 +0000233 int i;
Eric Andersencc8ed391999-10-05 16:24:54 +0000234
Eric Andersen29d2e361999-11-06 06:07:27 +0000235 /* Only compiled in if BB_MTAB is not defined */
236 whine_if_fstab_is_missing();
Eric Andersencb6e2561999-10-16 15:48:40 +0000237
Eric Andersen596e5461999-10-07 08:30:23 +0000238 if (argc == 1) {
Eric Andersend0246fb1999-11-04 21:18:07 +0000239 FILE *mountTable = setmntent (mtab_file, "r");
240 if (mountTable) {
Eric Andersen596e5461999-10-07 08:30:23 +0000241 struct mntent *m;
242 while ((m = getmntent (mountTable)) != 0) {
Eric Andersencb6e2561999-10-16 15:48:40 +0000243 struct fstab* fstabItem;
Eric Andersen596e5461999-10-07 08:30:23 +0000244 char *blockDevice = m->mnt_fsname;
Eric Andersencb6e2561999-10-16 15:48:40 +0000245 /* Note that if /etc/fstab is missing, libc can't fix up /dev/root for us */
246 if (strcmp (blockDevice, "/dev/root") == 0) {
247 fstabItem = getfsfile ("/");
248 if (fstabItem != NULL)
249 blockDevice = fstabItem->fs_spec;
250 }
Eric Andersen596e5461999-10-07 08:30:23 +0000251 printf ("%s on %s type %s (%s)\n", blockDevice, m->mnt_dir,
252 m->mnt_type, m->mnt_opts);
253 }
254 endmntent (mountTable);
Eric Andersend0246fb1999-11-04 21:18:07 +0000255 } else {
256 perror(mtab_file);
Eric Andersencc8ed391999-10-05 16:24:54 +0000257 }
Eric Andersen3c163821999-10-14 22:16:57 +0000258 exit( TRUE);
Eric Andersen596e5461999-10-07 08:30:23 +0000259 }
Eric Andersencc8ed391999-10-05 16:24:54 +0000260
Eric Andersen596e5461999-10-07 08:30:23 +0000261
262 /* Parse options */
Eric Andersen8341a151999-10-08 17:14:14 +0000263 i = --argc;
264 argv++;
265 while (i > 0 && **argv) {
Eric Andersen596e5461999-10-07 08:30:23 +0000266 if (**argv == '-') {
Eric Andersena9c95ea1999-11-15 17:33:30 +0000267 char *opt = *argv;
268 while (i>0 && *++opt) switch (*opt) {
Eric Andersen596e5461999-10-07 08:30:23 +0000269 case 'o':
Eric Andersen8341a151999-10-08 17:14:14 +0000270 if (--i == 0) {
Eric Andersend73dc5b1999-11-10 23:13:02 +0000271 goto goodbye;
Eric Andersencc8ed391999-10-05 16:24:54 +0000272 }
Eric Andersen8341a151999-10-08 17:14:14 +0000273 parse_mount_options (*(++argv), &flags, string_flags);
Eric Andersen596e5461999-10-07 08:30:23 +0000274 break;
275 case 'r':
276 flags |= MS_RDONLY;
277 break;
278 case 't':
Eric Andersen8341a151999-10-08 17:14:14 +0000279 if (--i == 0) {
Eric Andersend73dc5b1999-11-10 23:13:02 +0000280 goto goodbye;
Eric Andersen596e5461999-10-07 08:30:23 +0000281 }
Eric Andersen8341a151999-10-08 17:14:14 +0000282 filesystemType = *(++argv);
Eric Andersen596e5461999-10-07 08:30:23 +0000283 break;
284 case 'w':
285 flags &= ~MS_RDONLY;
286 break;
287 case 'a':
Eric Andersen3ae0c781999-11-04 01:13:21 +0000288 all = TRUE;
Eric Andersen596e5461999-10-07 08:30:23 +0000289 break;
Eric Andersend0246fb1999-11-04 21:18:07 +0000290#ifdef BB_MTAB
291 case 'f':
292 fakeIt = TRUE;
293 break;
294 case 'n':
295 useMtab = FALSE;
296 break;
297#endif
Eric Andersen596e5461999-10-07 08:30:23 +0000298 case 'v':
299 case 'h':
300 case '-':
Eric Andersend73dc5b1999-11-10 23:13:02 +0000301 goto goodbye;
Eric Andersen8341a151999-10-08 17:14:14 +0000302 }
303 } else {
304 if (device == NULL)
305 device=*argv;
306 else if (directory == NULL)
307 directory=*argv;
308 else {
Eric Andersend73dc5b1999-11-10 23:13:02 +0000309 goto goodbye;
Eric Andersen596e5461999-10-07 08:30:23 +0000310 }
311 }
312 i--;
313 argv++;
314 }
315
Eric Andersen3ae0c781999-11-04 01:13:21 +0000316 if (all == TRUE) {
Eric Andersen596e5461999-10-07 08:30:23 +0000317 struct mntent *m;
318 FILE *f = setmntent ("/etc/fstab", "r");
319
320 if (f == NULL) {
321 perror("/etc/fstab");
Eric Andersen3c163821999-10-14 22:16:57 +0000322 exit( FALSE);
Eric Andersen596e5461999-10-07 08:30:23 +0000323 }
Eric Andersen596e5461999-10-07 08:30:23 +0000324 while ((m = getmntent (f)) != NULL) {
Eric Andersen3ae0c781999-11-04 01:13:21 +0000325 // If the file system isn't noauto, and isn't mounted on /,
326 // and isn't swap or nfs, then mount it
327 if ((!strstr (m->mnt_opts, "noauto")) &&
328 (m->mnt_dir[1] != '\0') &&
329 (!strstr (m->mnt_type, "swap")) &&
330 (!strstr (m->mnt_type, "nfs")))
331 {
Eric Andersend0246fb1999-11-04 21:18:07 +0000332 flags = 0;
Eric Andersen3ae0c781999-11-04 01:13:21 +0000333 *string_flags = '\0';
Eric Andersend0246fb1999-11-04 21:18:07 +0000334 parse_mount_options(m->mnt_opts, &flags, string_flags);
335 mount_one (m->mnt_fsname, m->mnt_dir, m->mnt_type,
Eric Andersena9c95ea1999-11-15 17:33:30 +0000336 flags, string_flags, useMtab, fakeIt, extra_opts);
Eric Andersen596e5461999-10-07 08:30:23 +0000337 }
Eric Andersencc8ed391999-10-05 16:24:54 +0000338 }
Eric Andersen596e5461999-10-07 08:30:23 +0000339 endmntent (f);
340 } else {
Eric Andersen8341a151999-10-08 17:14:14 +0000341 if (device && directory) {
Eric Andersena9c95ea1999-11-15 17:33:30 +0000342#ifdef BB_NFSMOUNT
343 if (strcmp(filesystemType, "nfs") == 0) {
344 if (nfsmount(device, directory, &flags, &extra_opts, &string_flags, 1) != 0)
345 exit(FALSE);
346 }
347#endif
Eric Andersen3c163821999-10-14 22:16:57 +0000348 exit (mount_one (device, directory, filesystemType,
Eric Andersena9c95ea1999-11-15 17:33:30 +0000349 flags, string_flags, useMtab, fakeIt, extra_opts));
Eric Andersencc8ed391999-10-05 16:24:54 +0000350 } else {
Eric Andersend73dc5b1999-11-10 23:13:02 +0000351 goto goodbye;
Eric Andersencc8ed391999-10-05 16:24:54 +0000352 }
Eric Andersen596e5461999-10-07 08:30:23 +0000353 }
Eric Andersen3c163821999-10-14 22:16:57 +0000354 exit( TRUE);
Eric Andersend73dc5b1999-11-10 23:13:02 +0000355
356goodbye:
357 usage( mount_usage);
Eric Andersencc8ed391999-10-05 16:24:54 +0000358}