1 1.1 christos /* $NetBSD: umount_linux.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_linux.c 39 1.1 christos * 40 1.1 christos */ 41 1.1 christos 42 1.1 christos /* 43 1.1 christos * Linux method of unmounting filesystems: if umount(2) failed with EIO or 44 1.1 christos * ESTALE, try umount2(2) with MNT_FORCE+MNT_DETACH. 45 1.1 christos */ 46 1.1 christos 47 1.1 christos #ifdef HAVE_CONFIG_H 48 1.1 christos # include <config.h> 49 1.1 christos #endif /* HAVE_CONFIG_H */ 50 1.1 christos #include <am_defs.h> 51 1.1 christos #include <amu.h> 52 1.1 christos 53 1.1 christos 54 1.1 christos int 55 1.1 christos umount_fs(char *mntdir, const char *mnttabname, u_int unmount_flags) 56 1.1 christos { 57 1.1 christos mntlist *mlist, *mp, *mp_save = NULL; 58 1.1 christos int error = 0; 59 1.1 christos #ifdef HAVE_LOOP_DEVICE 60 1.1 christos char *opt, *xopts = NULL; 61 1.1 christos char loopstr[] = "loop="; 62 1.1 christos char *loopdev; 63 1.1 christos #endif /* HAVE_LOOP_DEVICE */ 64 1.1.1.3 christos unsigned int retries = 8; 65 1.1 christos 66 1.1 christos mp = mlist = read_mtab(mntdir, mnttabname); 67 1.1 christos 68 1.1 christos /* 69 1.1 christos * Search the mount table looking for 70 1.1 christos * the correct (ie last) matching entry 71 1.1 christos */ 72 1.1 christos while (mp) { 73 1.1 christos if (STREQ(mp->mnt->mnt_dir, mntdir)) 74 1.1 christos mp_save = mp; 75 1.1 christos mp = mp->mnext; 76 1.1 christos } 77 1.1 christos 78 1.1 christos if (!mp_save) { 79 1.1 christos plog(XLOG_ERROR, "Couldn't find how to unmount %s", mntdir); 80 1.1 christos /* Assume it is already unmounted */ 81 1.1 christos error = 0; 82 1.1 christos goto out; 83 1.1 christos } 84 1.1 christos 85 1.1.1.3 christos plog(XLOG_ERROR, "Trying unmount %s, umount_flags 0x%x", mp_save->mnt->mnt_dir, unmount_flags); 86 1.1 christos dlog("Trying unmount(%s)", mp_save->mnt->mnt_dir); 87 1.1 christos 88 1.1 christos #ifdef MOUNT_TABLE_ON_FILE 89 1.1 christos /* 90 1.1 christos * This unmount may hang leaving this process with an exclusive lock on 91 1.1 christos * /etc/mtab. Therefore it is necessary to unlock mtab, do the unmount, 92 1.1 christos * then lock mtab (again) and reread it and finally update it. 93 1.1 christos */ 94 1.1 christos unlock_mntlist(); 95 1.1 christos #endif /* MOUNT_TABLE_ON_FILE */ 96 1.1 christos 97 1.1.1.3 christos again: 98 1.1 christos #if defined(HAVE_UMOUNT2) && defined(MNT2_GEN_OPT_DETACH) 99 1.1 christos /* 100 1.1 christos * If user asked to try forced unmounts, then do a quick check to see if 101 1.1 christos * the mount point is hung badly. If so, then try to detach it by 102 1.1 christos * force; if the latter works, we're done. 103 1.1 christos */ 104 1.1 christos if (unmount_flags & AMU_UMOUNT_DETACH) { 105 1.1 christos /* 106 1.1 christos * Note: we pass both DETACH and FORCE flags, because umount2_fs below 107 1.1 christos * (on Linux), should try FORCE before DETACH (the latter always 108 1.1 christos * succeeds). 109 1.1 christos */ 110 1.1 christos error = umount2_fs(mp_save->mnt->mnt_dir, 111 1.1 christos unmount_flags & (AMU_UMOUNT_DETACH|AMU_UMOUNT_FORCE)); 112 1.1 christos } else 113 1.1 christos #endif /* defined(HAVE_UMOUNT2) && defined(MNT2_GEN_OPT_DETACH) */ 114 1.1 christos error = UNMOUNT_TRAP(mp_save->mnt); 115 1.1.1.3 christos 116 1.1.1.3 christos /* Linux kernel can be sluggish for some reason */ 117 1.1.1.3 christos if (error == EBUSY && retries--) { 118 1.1.1.3 christos struct timespec tm = {0, 200000000}; 119 1.1.1.3 christos nanosleep(&tm, NULL); 120 1.1.1.3 christos goto again; 121 1.1.1.3 christos } 122 1.1.1.3 christos 123 1.1 christos if (error < 0) { 124 1.1 christos plog(XLOG_WARNING, "unmount(%s) failed: %m", mp_save->mnt->mnt_dir); 125 1.1 christos switch ((error = errno)) { 126 1.1 christos case EINVAL: 127 1.1 christos case ENOTBLK: 128 1.1 christos plog(XLOG_WARNING, "unmount: %s is not mounted", mp_save->mnt->mnt_dir); 129 1.1 christos error = 0; /* Not really an error */ 130 1.1 christos break; 131 1.1 christos 132 1.1 christos case ENOENT: 133 1.1 christos /* 134 1.1 christos * This could happen if the kernel insists on following symlinks 135 1.1 christos * when we try to unmount a direct mountpoint. We need to propagate 136 1.1 christos * the error up so that the top layers know it failed and don't 137 1.1 christos * try to rmdir() the mountpoint or other silly things. 138 1.1 christos */ 139 1.1 christos plog(XLOG_ERROR, "mount point %s: %m", mp_save->mnt->mnt_dir); 140 1.1 christos break; 141 1.1 christos 142 1.1 christos #if defined(HAVE_UMOUNT2) && defined(MNT2_GEN_OPT_FORCE) 143 1.1 christos case EBUSY: 144 1.1 christos /* 145 1.1 christos * Caller determines if forced unmounts should be used now (for 146 1.1 christos * EBUSY). If caller asked to force an unmount, *and* the above 147 1.1 christos * "trivial" unmount attempt failed with EBUSY, then try to force 148 1.1 christos * the unmount. 149 1.1 christos */ 150 1.1 christos if (unmount_flags & AMU_UMOUNT_FORCE) { 151 1.1 christos error = umount2_fs(mp_save->mnt->mnt_dir, 152 1.1 christos unmount_flags & AMU_UMOUNT_FORCE); 153 1.1 christos if (error < 0) { 154 1.1 christos plog(XLOG_WARNING, "%s: unmount/force: %m", 155 1.1 christos mp_save->mnt->mnt_dir); 156 1.1 christos error = errno; 157 1.1 christos } 158 1.1 christos } 159 1.1 christos break; 160 1.1 christos #endif /* defined(HAVE_UMOUNT2) && defined(MNT2_GEN_OPT_FORCE) */ 161 1.1 christos 162 1.1 christos default: 163 1.1 christos dlog("%s: unmount: %m", mp_save->mnt->mnt_dir); 164 1.1 christos break; 165 1.1 christos } 166 1.1 christos } else { 167 1.1 christos dlog("unmount(%s) succeeded", mp_save->mnt->mnt_dir); 168 1.1 christos } 169 1.1 christos dlog("Finished unmount(%s)", mp_save->mnt->mnt_dir); 170 1.1 christos 171 1.1 christos /* 172 1.1 christos * If we are successful or there was an ENOENT, remove 173 1.1 christos * the mount entry from the mtab file. 174 1.1 christos */ 175 1.1 christos if (error && error != ENOENT) 176 1.1 christos goto out; 177 1.1 christos 178 1.1 christos #ifdef HAVE_LOOP_DEVICE 179 1.1 christos /* look for loop=/dev/loopX in mnt_opts */ 180 1.1.1.3 christos xopts = xstrdup(mp_save->mnt->mnt_opts); /* b/c strtok is destructive */ 181 1.1 christos for (opt = strtok(xopts, ","); opt; opt = strtok(NULL, ",")) 182 1.1 christos if (NSTREQ(opt, loopstr, sizeof(loopstr) - 1)) { 183 1.1 christos loopdev = opt + sizeof(loopstr) - 1; 184 1.1 christos if (delete_loop_device(loopdev) < 0) 185 1.1 christos plog(XLOG_WARNING, "unmount() failed to release loop device %s: %m", loopdev); 186 1.1 christos else 187 1.1 christos plog(XLOG_INFO, "unmount() released loop device %s OK", loopdev); 188 1.1 christos break; 189 1.1 christos } 190 1.1 christos if (xopts) 191 1.1 christos XFREE(xopts); 192 1.1 christos #endif /* HAVE_LOOP_DEVICE */ 193 1.1 christos 194 1.1 christos #ifdef MOUNT_TABLE_ON_FILE 195 1.1 christos free_mntlist(mlist); 196 1.1 christos mp = mlist = read_mtab(mntdir, mnttabname); 197 1.1 christos 198 1.1 christos /* 199 1.1 christos * Search the mount table looking for 200 1.1 christos * the correct (ie last) matching entry 201 1.1 christos */ 202 1.1 christos mp_save = NULL; 203 1.1 christos while (mp) { 204 1.1 christos if (STREQ(mp->mnt->mnt_dir, mntdir)) 205 1.1 christos mp_save = mp; 206 1.1 christos mp = mp->mnext; 207 1.1 christos } 208 1.1 christos 209 1.1 christos if (mp_save) { 210 1.1 christos mnt_free(mp_save->mnt); 211 1.1 christos mp_save->mnt = NULL; 212 1.1 christos rewrite_mtab(mlist, mnttabname); 213 1.1 christos } 214 1.1 christos #endif /* MOUNT_TABLE_ON_FILE */ 215 1.1 christos 216 1.1 christos out: 217 1.1 christos free_mntlist(mlist); 218 1.1 christos 219 1.1 christos return error; 220 1.1 christos } 221 1.1 christos 222 1.1 christos 223 1.1 christos #if defined(HAVE_UMOUNT2) && (defined(MNT2_GEN_OPT_FORCE) || defined(MNT2_GEN_OPT_DETACH)) 224 1.1 christos /* 225 1.1 christos * Force unmount, no questions asked, without touching mnttab file. Try 226 1.1 christos * detach first because it is safer: will remove the hung mnt point without 227 1.1 christos * affecting hung applications. "Force" is more risky: it will cause the 228 1.1 christos * kernel to return EIO to applications stuck on a stat(2) of Amd. 229 1.1 christos */ 230 1.1 christos int 231 1.1 christos umount2_fs(const char *mntdir, u_int unmount_flags) 232 1.1 christos { 233 1.1 christos int error = 0; 234 1.1 christos 235 1.1 christos #ifdef MNT2_GEN_OPT_DETACH 236 1.1 christos if (unmount_flags & AMU_UMOUNT_DETACH) { 237 1.1 christos error = umount2(mntdir, MNT2_GEN_OPT_DETACH); 238 1.1 christos if (error < 0 && (errno == EINVAL || errno == ENOENT)) 239 1.1 christos error = 0; /* ignore EINVAL/ENOENT */ 240 1.1 christos if (error < 0) { /* don't try FORCE if detach succeeded */ 241 1.1 christos plog(XLOG_WARNING, "%s: unmount/detach: %m", mntdir); 242 1.1 christos /* fall through to try "force" (if flag specified) */ 243 1.1 christos } else { 244 1.1 christos dlog("%s: unmount/detach: OK", mntdir); 245 1.1 christos return error; 246 1.1 christos } 247 1.1 christos } 248 1.1 christos #endif /* MNT2_GEN_OPT_DETACH */ 249 1.1 christos 250 1.1 christos #ifdef MNT2_GEN_OPT_FORCE 251 1.1 christos if (unmount_flags & AMU_UMOUNT_FORCE) { 252 1.1 christos plog(XLOG_INFO, "umount2_fs: trying unmount/forced on %s", mntdir); 253 1.1 christos error = umount2(mntdir, MNT2_GEN_OPT_FORCE); 254 1.1 christos if (error < 0 && (errno == EINVAL || errno == ENOENT)) 255 1.1 christos error = 0; /* ignore EINVAL/ENOENT */ 256 1.1 christos if (error < 0) 257 1.1 christos plog(XLOG_WARNING, "%s: unmount/force: %m", mntdir); 258 1.1 christos else 259 1.1 christos dlog("%s: unmount/force: OK", mntdir); 260 1.1 christos /* fall through to return whatever error we got (if any) */ 261 1.1 christos } 262 1.1 christos #endif /* MNT2_GEN_OPT_FORCE */ 263 1.1 christos 264 1.1 christos return error; 265 1.1 christos } 266 1.1 christos #endif /* HAVE_UMOUNT2 && (MNT2_GEN_OPT_FORCE || MNT2_GEN_OPT_DETACH) */ 267