/* ----------------------------------------------------------------------- * * * mount_ext2.c - module for Linux automountd to mount ext2 filesystems * after running fsck on them. * * Copyright 1998 Transmeta Corporation - All Rights Reserved * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, Inc., 675 Mass Ave, Cambridge MA 02139, * USA; either version 2 of the License, or (at your option) any later * version; incorporated herein by reference. * * ----------------------------------------------------------------------- */ #include #include #include #include #include #include #include #define MODULE_MOUNT #include "automount.h" #define MODPREFIX "mount(ext2): " int mount_version = AUTOFS_MOUNT_VERSION; /* Required by protocol */ int mount_init(void **context) { return 0; } int mount_reinit(void **context) { return 0; } int mount_mount(struct autofs_point *ap, const char *root, const char *name, int name_len, const char *what, const char *fstype, const char *options, void *context) { char fullpath[PATH_MAX]; char buf[MAX_ERR_BUF]; const char *p, *p1; int err, ro = 0; const char *fsck_prog; int len, status, existed = 1; if (ap->flags & MOUNT_FLAG_REMOUNT) return 0; /* Root offset of multi-mount */ len = strlen(root); if (root[len - 1] == '/') { len = snprintf(fullpath, len, "%s", root); } else if (*name == '/') { /* * Direct or offset mount, name is absolute path so * don't use root (but with move mount changes root * is now the same as name). */ len = sprintf(fullpath, "%s", root); } else { len = sprintf(fullpath, "%s/%s", root, name); } fullpath[len] = '\0'; debug(ap->logopt, MODPREFIX "calling mkdir_path %s", fullpath); status = mkdir_path(fullpath, mp_mode); if (status && errno != EEXIST) { char *estr = strerror_r(errno, buf, MAX_ERR_BUF); error(ap->logopt, MODPREFIX "mkdir_path %s failed: %s", fullpath, estr); return 1; } if (!status) existed = 0; if (options && options[0]) { for (p = options; (p1 = strchr(p, ',')); p = p1) if (!_strncmp("ro", p, p1 - p) && ++p1 - p == sizeof("ro")) ro = 1; if (!strcmp(p, "ro")) ro = 1; } fsck_prog = PATH_E2FSCK; #ifdef HAVE_E3FSCK if (!strcmp(fstype,"ext3")) fsck_prog = PATH_E3FSCK; #endif #ifdef HAVE_E4FSCK if (!strcmp(fstype,"ext4")) fsck_prog = PATH_E4FSCK; #endif if (ro) { debug(ap->logopt, MODPREFIX "calling %s -n %s", fsck_prog, what); err = spawnl(ap->logopt, fsck_prog, fsck_prog, "-n", what, NULL); } else { debug(ap->logopt, MODPREFIX "calling %s -p %s", fsck_prog, what); err = spawnl(ap->logopt, fsck_prog, fsck_prog, "-p", what, NULL); } /* * spawnl returns the error code, left shifted by 8 bits. We are * interested in the following error bits from the fsck program: * 2 - File system errors corrected, system should be rebooted * 4 - File system errors left uncorrected */ if ((err >> 8) & 6) { error(ap->logopt, MODPREFIX "%s: filesystem needs repair, won't mount", what); return 1; } if (options) { debug(ap->logopt, MODPREFIX "calling mount -t %s -o %s %s %s", fstype, options, what, fullpath); err = spawn_mount(ap->logopt, "-t", fstype, "-o", options, what, fullpath, NULL); } else { debug(ap->logopt, MODPREFIX "calling mount -t %s %s %s", fstype, what, fullpath); err = spawn_mount(ap->logopt, "-t", fstype, what, fullpath, NULL); } if (err) { info(ap->logopt, MODPREFIX "failed to mount %s (type %s) on %s", what, fstype, fullpath); if (ap->type != LKP_INDIRECT) return 1; if ((!(ap->flags & MOUNT_FLAG_GHOST) && name_len) || !existed) rmdir_path(ap, fullpath, ap->dev); return 1; } else { debug(ap->logopt, MODPREFIX "mounted %s type %s on %s", what, fstype, fullpath); return 0; } } int mount_done(void *context) { return 0; }