1 1.1 christos /* $NetBSD: umount_default.c,v 1.1.1.3 2015/01/17 16:34:16 christos Exp $ */ 2 1.1 christos 3 1.1 christos /* 4 1.1.1.3 christos * Copyright (c) 1997-2014 Erez Zadok 5 1.1 christos * Copyright (c) 1990 Jan-Simon Pendry 6 1.1 christos * Copyright (c) 1990 Imperial College of Science, Technology & Medicine 7 1.1 christos * Copyright (c) 1990 The Regents of the University of California. 8 1.1 christos * All rights reserved. 9 1.1 christos * 10 1.1 christos * This code is derived from software contributed to Berkeley by 11 1.1 christos * Jan-Simon Pendry at Imperial College, London. 12 1.1 christos * 13 1.1 christos * Redistribution and use in source and binary forms, with or without 14 1.1 christos * modification, are permitted provided that the following conditions 15 1.1 christos * are met: 16 1.1 christos * 1. Redistributions of source code must retain the above copyright 17 1.1 christos * notice, this list of conditions and the following disclaimer. 18 1.1 christos * 2. Redistributions in binary form must reproduce the above copyright 19 1.1 christos * notice, this list of conditions and the following disclaimer in the 20 1.1 christos * documentation and/or other materials provided with the distribution. 21 1.1.1.3 christos * 3. Neither the name of the University nor the names of its contributors 22 1.1 christos * may be used to endorse or promote products derived from this software 23 1.1 christos * without specific prior written permission. 24 1.1 christos * 25 1.1 christos * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 26 1.1 christos * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 27 1.1 christos * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 28 1.1 christos * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 29 1.1 christos * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 30 1.1 christos * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 31 1.1 christos * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 32 1.1 christos * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 33 1.1 christos * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 34 1.1 christos * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 35 1.1 christos * SUCH DAMAGE. 36 1.1 christos * 37 1.1 christos * 38 1.1 christos * File: am-utils/conf/umount/umount_default.c 39 1.1 christos * 40 1.1 christos */ 41 1.1 christos 42 1.1 christos /* 43 1.1 christos * Default method of unmounting filesystems. 44 1.1 christos */ 45 1.1 christos 46 1.1 christos #ifdef HAVE_CONFIG_H 47 1.1 christos # include <config.h> 48 1.1 christos #endif /* HAVE_CONFIG_H */ 49 1.1 christos #include <am_defs.h> 50 1.1 christos #include <amu.h> 51 1.1 christos 52 1.1 christos 53 1.1 christos int 54 1.1 christos umount_fs(char *mntdir, const char *mnttabname, u_int unmount_flags) 55 1.1 christos { 56 1.1 christos mntlist *mlist, *mp, *mp_save = NULL; 57 1.1 christos int error = 0; 58 1.1 christos 59 1.1 christos mp = mlist = read_mtab(mntdir, mnttabname); 60 1.1 christos 61 1.1 christos /* 62 1.1 christos * Search the mount table looking for 63 1.1 christos * the correct (ie last) matching entry 64 1.1 christos */ 65 1.1 christos while (mp) { 66 1.1 christos if (STREQ(mp->mnt->mnt_dir, mntdir)) 67 1.1 christos mp_save = mp; 68 1.1 christos mp = mp->mnext; 69 1.1 christos } 70 1.1 christos 71 1.1 christos if (mp_save) { 72 1.1 christos dlog("Trying unmount(%s)", mp_save->mnt->mnt_dir); 73 1.1 christos 74 1.1 christos #ifdef MOUNT_TABLE_ON_FILE 75 1.1 christos /* 76 1.1 christos * This unmount may hang leaving this process with an exclusive lock on 77 1.1 christos * /etc/mtab. Therefore it is necessary to unlock mtab, do the unmount, 78 1.1 christos * then lock mtab (again) and reread it and finally update it. 79 1.1 christos */ 80 1.1 christos unlock_mntlist(); 81 1.1 christos #endif /* MOUNT_TABLE_ON_FILE */ 82 1.1 christos 83 1.1 christos #ifdef NEED_AUTOFS_SPACE_HACK 84 1.1 christos if (unmount_flags & AMU_UMOUNT_AUTOFS) { 85 1.1 christos char *mnt_dir_save = mp_save->mnt->mnt_dir; 86 1.1 christos mp_save->mnt->mnt_dir = autofs_strdup_space_hack(mnt_dir_save); 87 1.1 christos error = UNMOUNT_TRAP(mp_save->mnt); 88 1.1 christos XFREE(mp_save->mnt->mnt_dir); 89 1.1 christos mp_save->mnt->mnt_dir = mnt_dir_save; 90 1.1 christos } else 91 1.1 christos #endif /* NEED_AUTOFS_SPACE_HACK */ 92 1.1 christos error = UNMOUNT_TRAP(mp_save->mnt); 93 1.1 christos if (error < 0) { 94 1.1 christos switch ((error = errno)) { 95 1.1 christos case EINVAL: 96 1.1 christos case ENOTBLK: 97 1.1 christos plog(XLOG_WARNING, "unmount: %s is not mounted", mp_save->mnt->mnt_dir); 98 1.1 christos error = 0; /* Not really an error */ 99 1.1 christos break; 100 1.1 christos 101 1.1 christos case ENOENT: 102 1.1 christos /* 103 1.1 christos * This could happen if the kernel insists on following symlinks 104 1.1 christos * when we try to unmount a direct mountpoint. We need to propagate 105 1.1 christos * the error up so that the top layers know it failed and don't 106 1.1 christos * try to rmdir() the mountpoint or other silly things. 107 1.1 christos */ 108 1.1 christos plog(XLOG_ERROR, "mount point %s: %m", mp_save->mnt->mnt_dir); 109 1.1 christos break; 110 1.1 christos 111 1.1 christos #if defined(MNT2_GEN_OPT_FORCE) && defined(HAVE_UMOUNT2) 112 1.1 christos case EBUSY: 113 1.1 christos case EIO: 114 1.1 christos case ESTALE: 115 1.1 christos /* caller determines if forced unmounts should be used */ 116 1.1 christos if (unmount_flags & AMU_UMOUNT_FORCE) { 117 1.1 christos error = umount2_fs(mntdir, unmount_flags); 118 1.1 christos if (error < 0) 119 1.1 christos error = errno; 120 1.1 christos else 121 1.1 christos break; /* all is OK */ 122 1.1 christos } 123 1.1 christos /* fallthrough */ 124 1.1 christos #endif /* MNT2_GEN_OPT_FORCE && HAVE_UMOUNT2 */ 125 1.1 christos default: 126 1.1 christos dlog("%s: unmount: %m", mp_save->mnt->mnt_dir); 127 1.1 christos break; 128 1.1 christos } 129 1.1 christos } 130 1.1 christos dlog("Finished unmount(%s)", mp_save->mnt->mnt_dir); 131 1.1 christos 132 1.1 christos if (!error) { 133 1.1 christos #ifdef MOUNT_TABLE_ON_FILE 134 1.1 christos free_mntlist(mlist); 135 1.1 christos mp = mlist = read_mtab(mntdir, mnttabname); 136 1.1 christos 137 1.1 christos /* 138 1.1 christos * Search the mount table looking for 139 1.1 christos * the correct (ie last) matching entry 140 1.1 christos */ 141 1.1 christos mp_save = NULL; 142 1.1 christos while (mp) { 143 1.1 christos if (STREQ(mp->mnt->mnt_dir, mntdir)) 144 1.1 christos mp_save = mp; 145 1.1 christos mp = mp->mnext; 146 1.1 christos } 147 1.1 christos 148 1.1 christos if (mp_save) { 149 1.1 christos mnt_free(mp_save->mnt); 150 1.1 christos mp_save->mnt = NULL; 151 1.1 christos rewrite_mtab(mlist, mnttabname); 152 1.1 christos } 153 1.1 christos #endif /* MOUNT_TABLE_ON_FILE */ 154 1.1 christos } 155 1.1 christos 156 1.1 christos } else { 157 1.1 christos 158 1.1 christos plog(XLOG_ERROR, "Couldn't find how to unmount %s", mntdir); 159 1.1 christos /* 160 1.1 christos * Assume it is already unmounted 161 1.1 christos */ 162 1.1 christos error = 0; 163 1.1 christos } /* end of "if (mp_save)" statement */ 164 1.1 christos 165 1.1 christos free_mntlist(mlist); 166 1.1 christos 167 1.1 christos return error; 168 1.1 christos } 169 1.1 christos 170 1.1 christos 171 1.1 christos #if defined(MNT2_GEN_OPT_FORCE) && defined(HAVE_UMOUNT2) 172 1.1 christos /* force unmount, no questions asked, without touching mnttab file */ 173 1.1 christos int 174 1.1 christos umount2_fs(const char *mntdir, u_int unmount_flags) 175 1.1 christos { 176 1.1 christos int error = 0; 177 1.1 christos 178 1.1 christos if (unmount_flags & AMU_UMOUNT_FORCE) { 179 1.1 christos plog(XLOG_INFO, "umount2_fs: trying unmount/forced on %s", mntdir); 180 1.1 christos error = umount2(mntdir, MNT2_GEN_OPT_FORCE); /* Solaris */ 181 1.1 christos if (error < 0 && (errno == EINVAL || errno == ENOENT)) 182 1.1 christos error = 0; /* ignore EINVAL/ENOENT */ 183 1.1 christos if (error < 0) 184 1.1 christos plog(XLOG_WARNING, "%s: unmount/force: %m", mntdir); 185 1.1 christos else 186 1.1 christos dlog("%s: unmount/force: OK", mntdir); 187 1.1 christos } 188 1.1 christos return error; 189 1.1 christos } 190 1.1 christos #endif /* MNT2_GEN_OPT_FORCE && HAVE_UMOUNT2 */ 191