Set a more sensible default CPU share
[util-vserver.git] / src / secure-mount.c
1 // $Id: secure-mount.c,v 1.24 2005/03/24 12:45:06 ensc Exp $    --*- c++ -*--
2
3 // Copyright (C) 2003 Enrico Scholz <enrico.scholz@informatik.tu-chemnitz.de>
4 //  
5 // This program is free software; you can redistribute it and/or modify
6 // it under the terms of the GNU General Public License as published by
7 // the Free Software Foundation; version 2 of the License.
8 //  
9 // This program is distributed in the hope that it will be useful,
10 // but WITHOUT ANY WARRANTY; without even the implied warranty of
11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12 // GNU General Public License for more details.
13 //  
14 // You should have received a copy of the GNU General Public License
15 // along with this program; if not, write to the Free Software
16 // Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
17
18
19   // secure-mount <general mount(8) options> [--chroot]
20   //              [--mtab <mtabfile>] [--fstab <fstabfile>]
21   //
22   // Executes mount-operations under the current directory: it assumes sources
23   // in the current root-dir while destinations are expected in the chroot
24   // environment.
25
26
27 #ifdef HAVE_CONFIG_H
28 #  include <config.h>
29 #endif
30
31 #include "util.h"
32 #include "pathconfig.h"
33
34 #include <lib/internal.h>
35
36 #include <getopt.h>
37 #include <fcntl.h>
38 #include <errno.h>
39 #include <stdio.h>
40 #include <stdlib.h>
41 #include <string.h>
42 #include <strings.h>
43 #include <unistd.h>
44 #include <stdbool.h>
45 #include <sys/mount.h>
46 #include <sys/stat.h>
47 #include <sys/types.h>
48 #include <sys/file.h>
49 #include <linux/fs.h>
50 #include <assert.h>
51 #include <ctype.h>
52 #include <sys/wait.h>
53 #include <libgen.h>
54 #include <signal.h>
55
56 #define ENSC_WRAPPERS_FCNTL     1
57 #define ENSC_WRAPPERS_UNISTD    1
58 #include <wrappers.h>
59
60 #define MNTPOINT        "/etc"
61
62 typedef enum { rfsYES, rfsNO, rfsONLY }         RootFsOption;
63
64 struct MountInfo {
65     char const *        src;
66     char const *        dst;
67     char const *        type;
68     unsigned long       flag;
69     unsigned long       xflag;
70     unsigned long       mask;
71     char *              data;
72 };
73
74 struct Options {
75     char const *        mtab;
76     char const *        fstab;
77     bool                do_chroot;
78     bool                ignore_mtab;
79     bool                mount_all;
80     RootFsOption        rootfs;
81
82     int                 cur_dir_fd;
83     int                 cur_rootdir_fd;
84 };
85
86 #define OPTION_BIND     1024
87 #define OPTION_MOVE     1025
88 #define OPTION_MTAB     1026
89 #define OPTION_FSTAB    1027
90 #define OPTION_CHROOT   1028
91 #define OPTION_SECURE   1029
92 #define OPTION_RBIND    1030
93 #define OPTION_ROOTFS   1031
94
95 #define XFLAG_NOAUTO    0x01
96
97 static struct option const
98 CMDLINE_OPTIONS[] = {
99   { "help",    no_argument,       0, 'h' },
100   { "version", no_argument,       0, 'v' },
101   { "bind",    no_argument,       0, OPTION_BIND },
102   { "move",    no_argument,       0, OPTION_MOVE },
103   { "mtab",    required_argument, 0, OPTION_MTAB },
104   { "fstab",   required_argument, 0, OPTION_FSTAB },
105   { "rootfs",  required_argument, 0, OPTION_ROOTFS },
106   { "chroot",  no_argument,       0, OPTION_CHROOT },
107   { "secure",  no_argument,       0, OPTION_SECURE },
108   { "rbind",   no_argument,       0, OPTION_RBIND },
109   { 0, 0, 0, 0 }
110 };
111
112 #ifndef MS_REC
113 #  define MS_REC        0x4000
114 #endif
115
116 static struct FstabOption {
117     char const * const  opt;
118     unsigned long const         flag;
119     unsigned long const         mask;
120     unsigned long const         xflag;
121     bool const                  is_dflt;
122 } const FSTAB_OPTIONS[] = {
123   { "defaults",   0,             (MS_RDONLY|MS_NOSUID|MS_NODEV|MS_NOEXEC|
124                                   MS_SYNCHRONOUS), 0, false },
125   { "rbind",      MS_BIND|MS_REC, MS_BIND|MS_REC,  0, false },
126   { "bind",       MS_BIND,        MS_BIND,         0, false },
127   { "move",       MS_MOVE,        MS_MOVE,         0, false },
128   { "async",      0,              MS_SYNCHRONOUS,  0, false },
129   { "sync",       MS_SYNCHRONOUS, MS_SYNCHRONOUS,  0, false },
130   { "atime",      0,              MS_NOATIME,      0, false },
131   { "noatime",    MS_NOATIME,     MS_NOATIME,      0, false },
132   { "dev",        0,              MS_NODEV,        0, false },
133   { "nodev",      MS_NODEV,       MS_NODEV,        0, false },
134   { "exec",       0,              MS_NOEXEC,       0, false },
135   { "noexec",     MS_NOEXEC,      MS_NOEXEC,       0, false },
136   { "suid",       0,              MS_NOSUID,       0, false },
137   { "nosuid",     MS_NOSUID,      MS_NOSUID,       0, false },
138   { "ro",         MS_RDONLY,      MS_RDONLY,       0, false },
139   { "rw",         0,              MS_RDONLY,       0, false },
140   
141   { "remount",    MS_REMOUNT,     MS_REMOUNT,      0, false },
142   { "users",      MS_NOEXEC|MS_NOSUID|MS_NODEV,
143                   MS_NOEXEC|MS_NOSUID|MS_NODEV,    0, false },
144   { "mandlock",   MS_MANDLOCK,    MS_MANDLOCK,     0, false },
145   { "nodiratime", MS_NODIRATIME,  MS_NODIRATIME,   0, false },
146 #ifdef MS_DIRSYNC  
147   { "dirsync",    MS_DIRSYNC,     MS_DIRSYNC,      0, false },
148 #endif
149   { "_netdev",    0,              0,               0, false },
150   { "auto",       0,              0,               0, false },
151   { "noauto",     0,              0,               XFLAG_NOAUTO, false },
152   { "user",       0,              0,               0, false },
153   { "nouser",     0,              0,               0, false },
154 };
155
156 int                     wrapper_exit_code = 1;
157
158 static void
159 showHelp(int fd, char const *cmd, int res)
160 {
161   VSERVER_DECLARE_CMD(cmd);
162   
163   WRITE_MSG(fd, "Usage:  ");
164   WRITE_STR(fd, cmd);
165   WRITE_MSG(fd,
166             " [--help] [--version] [--bind] [--move] [--rbind] [-t <type>] [--chroot]\n"
167             "            [--mtab <filename>] [--fstab <filename>] [--rootfs yes|no|only]\n"
168             "            [-n] -a|([-o <options>] [--] <src> <dst>)\n\n"
169             "Executes mount-operations under the current directory: it assumes sources in\n"
170             "the current root-dir while destinations are expected in the chroot environment.\n\n"
171             "For non-trivial mount-operations it uses the external 'mount' program which\n"
172             "can be overridden by the $MOUNT environment variable.\n\n"
173             "Options:\n"
174             "  --bind|move|rbind        ...  set the correspond flags; with this options\n"
175             "                                the mount will be executed internally without\n"
176             "                                calling an external mount program.\n"
177             "  -t <type>                ...  assume the given filesystem type\n"
178             "  -o <options>             ...  set additional options; see mount(2) for details\n"
179             "  -n                       ...  do not update the mtab-file\n"
180             "  --mtab <filename>        ...  use <filename> as an alternative mtab file\n"
181             "                                [default: /etc/mtab]\n"
182             "  --chroot                 ...  chroot into the current directory before\n"
183             "                                mounting the filesystem\n"
184             "  --fstab <filename>       ...  use <filename> as an alternative fstab file;\n"
185             "                                this option has an effect only with the '-a'\n"
186             "                                option [default: /etc/fstab]\n"
187             "  --rootfs yes|no|only     ...  specifies how to handle an entry for a rootfs\n"
188             "                                ('/') when processing an fstab file. 'yes' will\n"
189             "                                mount it among the other entries, 'only' will\n"
190             "                                mount only the rootfs entry, and 'no' will ignore\n"
191             "                                it and mount only the other entries [default: yes]\n"
192             "  -a                       ...  mount everything listed in the fstab-file\n\n"
193             "  <src>                    ...  the source-filesystem; this path is absolute\n"
194             "                                to the current root-filesystem. Only valid\n"
195             "                                without the '-a' option.\n"
196             "  <dst>                    ...  the destination mount-point; when used with\n"
197             "                                '--chroot', this path is relative to the current\n"
198             "                                directory. Only valid without the '-a' option\n\n"
199             "Please report bugs to " PACKAGE_BUGREPORT "\n");
200
201   exit(res);
202 }
203
204 static void
205 showVersion()
206 {
207   WRITE_MSG(1,
208             "secure-mount " VERSION " -- secure mounting of directories\n"
209             "This program is part of " PACKAGE_STRING "\n\n"
210             "Copyright (C) 2003 Enrico Scholz\n"
211             VERSION_COPYRIGHT_DISCLAIMER);
212   exit(0);
213 }
214
215 inline static bool
216 isSameObject(struct stat const *lhs,
217              struct stat const *rhs)
218 {
219   return (lhs->st_dev==rhs->st_dev &&
220           lhs->st_ino==rhs->st_ino);
221 }
222
223 static int
224 fchroot(int fd)
225 {
226   if (fchdir(fd)==-1 || chroot(".")==-1) return -1;
227   return 0;
228 }
229
230 static int
231 writeX(int fd, void const *buf, size_t len)
232 {
233   if ((size_t)(write(fd, buf, len))!=len) return -1;
234   return 0;
235 }
236
237 static int
238 writeStrX(int fd, char const *str)
239 {
240   return writeX(fd, str, strlen(str));
241 }
242
243 static inline char const *
244 getType(struct MountInfo const *mnt)
245 {
246   if (mnt->type==0)                         return "none";
247   else if (strncmp(mnt->type, "ext", 3)==0) return "ufs";
248   else                                      return mnt->type;
249 }
250
251 inline static void
252 restoreRoot(struct Options const *opt)
253 {
254   if (opt->do_chroot!=0 && fchroot(opt->cur_rootdir_fd)==-1) {
255     perror("secure-mount: fchdir(\"/\")");
256     WRITE_MSG(2, "Failed to restore root-directory; aborting\n");
257     exit(1);
258   }
259 }
260
261 static int
262 updateMtab(struct MountInfo const *mnt, struct Options const *opt)
263 {
264   int           res = -1;
265   int           fd;
266   assert(opt->mtab!=0);
267
268   if (opt->do_chroot && fchroot(opt->cur_dir_fd)==-1) {
269       perror("secure-mount: fchroot(\".\")");
270       return -1;
271   }
272
273   fd=open(opt->mtab, O_CREAT|O_APPEND|O_WRONLY, 0644);
274   
275   if (fd==-1) {
276     perror("secure-mount: open(<mtab>)");
277     goto err0;
278   }
279
280   if (flock(fd, LOCK_EX)==-1) {
281     perror("secure-mount: flock()");
282     goto err1;
283   }
284
285   if (writeStrX(fd, mnt->src)==-1 ||
286       writeStrX(fd, " ")==-1 ||
287       writeStrX(fd, mnt->dst)==-1 ||
288       writeStrX(fd, " ")==-1 ||
289       writeStrX(fd, getType(mnt))==-1 ||
290       writeStrX(fd, " ")==-1 ||
291       writeStrX(fd, mnt->data ? mnt->data : "defaults")==-1 ||
292       writeStrX(fd, " 0 0\n")==-1) {
293     perror("secure-mount: write()");
294     goto err1;
295   }
296
297   res = 0;
298
299   err1: close(fd);
300   err0:
301   restoreRoot(opt);
302   return res;
303 }
304
305 static bool
306 callExternalMount(struct MountInfo const *mnt)
307 {
308   char const *  argv[10];
309   size_t        idx = 0;
310   pid_t         pid;
311   int           status;
312   char const *  mount_prog = getenv("MOUNT");
313
314   if (mount_prog==0) mount_prog = MOUNT_PROG;
315
316   argv[idx++] = mount_prog;
317   argv[idx++] = "-n";
318   if      (mnt->flag & MS_BIND) argv[idx++] = "--bind";
319   else if (mnt->flag & MS_MOVE) argv[idx++] = "--move";
320
321   argv[idx++] = "-o";
322   if (mnt->data && *mnt->data &&
323       strcmp(mnt->data, "defaults")!=0) {
324     if (mnt->mask & MS_NODEV)
325       argv[idx++] = mnt->data;
326     else {
327       char *    tmp = alloca(strlen(mnt->data) + sizeof("nodev,"));
328       strcpy(tmp, "nodev,");
329       strcat(tmp, mnt->data);
330       argv[idx++] = tmp;
331     }
332   }
333   else
334     argv[idx++] = "nodev";
335
336   if (mnt->type) {
337     argv[idx++] = "-t";
338     argv[idx++] = mnt->type;
339   }
340
341   argv[idx++] = mnt->src;
342   argv[idx++] = ".";
343   argv[idx]   = 0;
344
345   pid = fork();
346   if (pid==-1) {
347     perror("secure-mount: fork()");
348     return false;
349   }
350
351   if (pid==0) {
352     execv(mount_prog, const_cast(char **)(argv));
353     PERROR_Q("secure-mount: execv", mount_prog);
354     exit(1);
355   }
356
357   if (wait4(pid, &status, 0, 0)==-1) {
358     perror("secure-mount: wait4()");
359     return false;
360   }
361
362   return (WIFEXITED(status)) && (WEXITSTATUS(status)==0);
363 }
364
365 inline static bool
366 secureChdir(char const *dir, struct Options const *opt)
367 {
368   int           dir_fd;
369   bool          res = false;
370   
371   if (opt->do_chroot!=0 && fchroot(opt->cur_dir_fd)==-1) {
372     perror("secure-mount: fchroot(\".\")");
373     return false;
374   }
375
376   if (chdir(dir)==-1) {
377     PERROR_Q("secure-mount: chdir", dir);
378     goto err;
379   }
380
381   dir_fd = open(".", O_RDONLY|O_DIRECTORY);
382   if (dir_fd==-1) {
383     perror("secure-mount: open(\".\")");
384     goto err;
385   }
386
387   restoreRoot(opt);
388   if (fchdir(dir_fd)==-1)
389     PERROR_Q("secure-mount: fchdir", dir);
390   else
391     res = true;
392   
393   close(dir_fd);
394   return res;
395
396   err:
397   restoreRoot(opt);
398   return false;
399 }
400
401 static bool
402 mountSingle(struct MountInfo const *mnt, struct Options const *opt)
403 {
404   assert(mnt->dst!=0);
405   
406   if (!secureChdir(mnt->dst, opt))
407     return false;
408
409   if (mnt->flag & (MS_BIND|MS_MOVE)) {
410     unsigned long       flag = mnt->flag;
411     if ((flag & MS_NODEV)==0) flag |= MS_NODEV;
412     
413     if (mount(mnt->src, ".",
414               mnt->type ? mnt->type : "",
415               flag,  mnt->data)==-1) {
416       perror("secure-mount: mount()");
417       return false;
418     }
419   }
420   else if (!callExternalMount(mnt))
421     return false;
422
423   if (!opt->ignore_mtab &&
424       updateMtab(mnt, opt)==-1) {
425     WRITE_MSG(2, "Failed to update mtab-file\n");
426       // no error
427   }
428   
429   return true;
430 }
431
432 static struct FstabOption const *
433 searchOption(char const *opt, size_t len)
434 {
435   struct FstabOption const *            i;
436   for (i=FSTAB_OPTIONS+0; i<FSTAB_OPTIONS+DIM_OF(FSTAB_OPTIONS); ++i)
437     if (strncmp(i->opt, opt, len)==0) return i;
438
439   return 0;
440 }
441
442 static bool
443 transformOptionList(struct MountInfo *info, size_t UNUSED *col)
444 {
445   char const *                  ptr = info->data;
446
447   do {
448     char const *                pos = strchr(ptr, ',');
449     struct FstabOption const *  opt;
450     
451     if (pos==0) pos = ptr+strlen(ptr);
452     opt = searchOption(ptr, pos-ptr);
453
454     if (opt!=0) {
455       info->flag  &= ~opt->mask;
456       info->flag  |=  opt->flag;
457       info->mask  |=  opt->mask;
458       info->xflag |=  opt->xflag;
459     }
460
461     if (*pos!='\0')
462       ptr = pos+1;
463     else
464       ptr = pos;
465
466   } while (*ptr!='\0');
467
468   return true;
469 }
470
471 #define MOVE_TO_NEXT_FIELD(PTR,ALLOW_EOL)               \
472   while (!isspace(*PTR) && *PTR!='\0') ++PTR;           \
473   if (col) *col = buf-start_buf+1;                      \
474   if (!(ALLOW_EOL) && *PTR=='\0') return prFAIL;        \
475   *PTR++ = '\0';                                        \
476   while (isspace(*PTR)) ++PTR
477
478 static enum {prDOIT, prFAIL, prIGNORE}
479   parseFstabLine(struct MountInfo *info, char *buf, size_t *col)
480 {
481   char const * const    start_buf = buf;
482   size_t                err_col;
483
484   while (isspace(*buf)) ++buf;
485   if (*buf=='#' || *buf=='\0')  return prIGNORE;
486
487   info->src  = buf;
488   MOVE_TO_NEXT_FIELD(buf, false);
489   info->dst  = buf;
490   MOVE_TO_NEXT_FIELD(buf, false);
491   info->type = buf;
492   MOVE_TO_NEXT_FIELD(buf, false);
493   err_col    = buf-start_buf+1;
494   info->data = buf;
495   MOVE_TO_NEXT_FIELD(buf, true);
496
497   info->flag  = MS_NODEV;
498   info->mask  = 0;
499   info->xflag = 0;
500
501   if      (strcmp(info->type, "swap")  ==0) return prIGNORE;
502   else if (strcmp(info->type, "none")  ==0) info->type  = 0;
503   else if (strcmp(info->type, "devpts")==0) info->mask |= MS_NODEV;
504
505   if (col) *col = err_col;
506   if (!transformOptionList(info,col)) return prFAIL;
507   if (info->xflag & XFLAG_NOAUTO)     return prIGNORE;
508
509   return prDOIT;
510 }
511
512 #undef MOVE_TO_NEXT_FIELD
513
514 static void
515 showFstabPosition(int fd, char const *fname, size_t line_nr, size_t col_nr)
516 {
517   char          buf[3*sizeof(line_nr)*2 + 4];
518   size_t        len = utilvserver_fmt_uint(buf+1, line_nr)+1;
519   
520   buf[0]     = ':';
521   buf[len++] = ':';
522   len += utilvserver_fmt_uint(buf+len, col_nr);
523   WRITE_STR(fd, fname);
524   Vwrite(fd, buf, len);
525 }
526
527
528 static bool
529 mountFstab(struct Options const *opt)
530 {
531   bool          res = false;
532   int           fd;
533   off_t         len;
534
535   assert(opt->fstab!=0);
536   fd = open(opt->fstab, O_RDONLY);
537   if (fd==-1) {
538     perror("secure-mount: open(<fstab>)");
539     goto err0;
540   }
541
542   len = lseek(fd, 0, SEEK_END); 
543   if (len==-1 ||
544       lseek(fd, 0, SEEK_SET)==-1) {
545     perror("secure-mount: lseek(<fstab>)");
546     goto err1;
547   }
548
549   {
550     char        buf[len+2];
551     char        *ptr, *ptrptr;
552     size_t      line_nr=0, col_nr;
553
554     if (read(fd, buf, len+1)!=len) {
555       perror("secure-mount: read()");
556       goto err1;
557     }
558     buf[len]   = '#';   // workaround for broken dietlibc strtok_r()
559                         // implementation
560     buf[len+1] = '\0';
561     ptrptr     = buf;
562
563     while ((ptr=strsep(&ptrptr, "\n")) != 0) {
564       struct MountInfo  mnt;
565       ++line_nr;
566
567       switch (parseFstabLine(&mnt, ptr, &col_nr)) {
568         case prFAIL     :
569           showFstabPosition(2, opt->fstab, line_nr, col_nr);
570           WRITE_MSG(2, ": syntax error\n");
571           goto err1;
572
573         case prIGNORE   :  break;
574         case prDOIT     : {
575           bool          is_rootfs = (strcmp(mnt.dst, "/")==0);
576           Echdir("/");
577           if (( is_rootfs && opt->rootfs==rfsNO) ||
578               (!is_rootfs && opt->rootfs==rfsONLY)) { /* ignore the entry */ }
579           else if (!mountSingle(&mnt, opt)) {
580             showFstabPosition(2, opt->fstab, line_nr, 1);
581             WRITE_MSG(2, ": failed to mount fstab-entry\n");
582           }
583           break;
584         }
585         default         :
586           assert(false);
587       }
588     }
589   }
590
591   res = true;
592
593   err1: close(fd);
594   err0: return res;
595 }
596
597 static void
598 initFDs(struct Options *opt)
599 {
600   opt->cur_dir_fd     = Eopen(".", O_RDONLY|O_DIRECTORY, 0);
601   opt->cur_rootdir_fd = Eopen("/", O_RDONLY|O_DIRECTORY, 0);
602
603   Efcntl(opt->cur_dir_fd,     F_SETFD, FD_CLOEXEC);
604   Efcntl(opt->cur_rootdir_fd, F_SETFD, FD_CLOEXEC);
605 }
606
607 static RootFsOption
608 parseRootFS(char const *str)
609 {
610   if      (strcasecmp(str, "yes")==0)  return rfsYES;
611   else if (strcasecmp(str, "no")==0)   return rfsNO;
612   else if (strcasecmp(str, "only")==0) return rfsONLY;
613   else {
614     WRITE_MSG(2, "secure-mount: invalid option for '--rootfs': '");
615     WRITE_STR(2, str);
616     WRITE_MSG(2, "'\n");
617     exit(wrapper_exit_code);
618   }
619 }
620
621 int main(int argc, char *argv[])
622 {
623   struct MountInfo      mnt = {
624     .src         = 0,
625     .dst         = 0,
626     .type        = 0,
627     .flag        = 0,
628     .xflag       = 0,
629     .data        = 0,
630   };
631
632   struct Options        opt = {
633     .mtab           = "/etc/mtab",
634     .fstab          = "/etc/fstab",
635     .do_chroot      = 0,
636     .ignore_mtab    = false,
637     .mount_all      = false,
638     .cur_dir_fd     = -1,
639     .cur_rootdir_fd = -1,
640     .rootfs         = rfsYES
641   };
642
643   while (1) {
644     int         c = getopt_long(argc, argv, "ht:nao:", CMDLINE_OPTIONS, 0);
645     if (c==-1) break;
646     
647     switch (c) {
648       case 'h'          :  showHelp(1, argv[0], 0);
649       case 'v'          :  showVersion();
650       case 't'          :  mnt.type = optarg;         break;
651       case 'n'          :  opt.ignore_mtab = true;    break;
652       case 'a'          :  opt.mount_all   = true;    break;
653       case 'o'          :  mnt.data        = optarg;  break;
654       case OPTION_RBIND :  mnt.flag       |= MS_REC;  /*@fallthrough@*/
655       case OPTION_BIND  :  mnt.flag       |= MS_BIND; break;
656       case OPTION_MOVE  :  mnt.flag       |= MS_MOVE; break;
657       case OPTION_MTAB  :  opt.mtab        = optarg;  break;
658       case OPTION_FSTAB :  opt.fstab       = optarg;  break;
659       case OPTION_CHROOT:  opt.do_chroot   = true;    break;
660       case OPTION_ROOTFS:  opt.rootfs      = parseRootFS(optarg); break;
661       case OPTION_SECURE:
662         WRITE_MSG(2, "secure-mount: The '--secure' option is deprecated...\n");
663         break;
664       default           :
665         WRITE_MSG(2, "Try '");
666         WRITE_STR(2, argv[0]);
667         WRITE_MSG(2, " --help\" for more information.\n");
668         return EXIT_FAILURE;
669         break;
670     }
671   }
672
673
674   if (opt.mount_all && optind<argc) {
675     WRITE_MSG(2, "Can not specify <src> and '-a' at the same time\n");
676     return EXIT_FAILURE;
677   }
678
679   initFDs(&opt);
680   signal(SIGCHLD, SIG_DFL);
681   
682   if (opt.mount_all) {
683     if (!mountFstab(&opt)) return EXIT_FAILURE;
684     else                   return EXIT_SUCCESS;
685   }
686
687   if (optind+2!=argc) {
688     WRITE_MSG(2, "Invalid <src> <dst> pair specified\n");
689     return EXIT_FAILURE;
690   }
691
692   if (mnt.data) {
693     mnt.data = strdup(mnt.data);
694     if (!transformOptionList(&mnt, 0)) {
695       WRITE_MSG(2, "Invalid options specified\n");
696       return EXIT_FAILURE;
697     }
698   }
699     
700   mnt.src  = argv[optind++];
701   mnt.dst  = argv[optind++];
702
703   if (!mountSingle(&mnt, &opt)) return EXIT_FAILURE;
704     
705   return EXIT_SUCCESS;
706 }