4 * Copyright (c) 1997-2009 Erez Zadok
5 * Copyright (c) 1990 Jan-Simon Pendry
6 * Copyright (c) 1990 Imperial College of Science, Technology & Medicine
7 * Copyright (c) 1990 The Regents of the University of California.
10 * This code is derived from software contributed to Berkeley by
11 * Jan-Simon Pendry at Imperial College, London.
13 * Redistribution and use in source and binary forms, with or without
14 * modification, are permitted provided that the following conditions
16 * 1. Redistributions of source code must retain the above copyright
17 * notice, this list of conditions and the following disclaimer.
18 * 2. Redistributions in binary form must reproduce the above copyright
19 * notice, this list of conditions and the following disclaimer in the
20 * documentation and/or other materials provided with the distribution.
21 * 3. All advertising materials mentioning features or use of this software
22 * must display the following acknowledgment:
23 * This product includes software developed by the University of
24 * California, Berkeley and its contributors.
25 * 4. Neither the name of the University nor the names of its contributors
26 * may be used to endorse or promote products derived from this software
27 * without specific prior written permission.
29 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
30 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
31 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
32 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
33 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
34 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
35 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
36 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
37 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
38 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
42 * File: am-utils/conf/umount/umount_default.c
47 * Default method of unmounting filesystems.
52 #endif /* HAVE_CONFIG_H */
58 umount_fs(char *mntdir
, const char *mnttabname
, u_int unmount_flags
)
60 mntlist
*mlist
, *mp
, *mp_save
= NULL
;
63 mp
= mlist
= read_mtab(mntdir
, mnttabname
);
66 * Search the mount table looking for
67 * the correct (ie last) matching entry
70 if (STREQ(mp
->mnt
->mnt_dir
, mntdir
))
76 dlog("Trying unmount(%s)", mp_save
->mnt
->mnt_dir
);
78 #ifdef MOUNT_TABLE_ON_FILE
80 * This unmount may hang leaving this process with an exclusive lock on
81 * /etc/mtab. Therefore it is necessary to unlock mtab, do the unmount,
82 * then lock mtab (again) and reread it and finally update it.
85 #endif /* MOUNT_TABLE_ON_FILE */
87 #ifdef NEED_AUTOFS_SPACE_HACK
88 if (unmount_flags
& AMU_UMOUNT_AUTOFS
) {
89 char *mnt_dir_save
= mp_save
->mnt
->mnt_dir
;
90 mp_save
->mnt
->mnt_dir
= autofs_strdup_space_hack(mnt_dir_save
);
91 error
= UNMOUNT_TRAP(mp_save
->mnt
);
92 XFREE(mp_save
->mnt
->mnt_dir
);
93 mp_save
->mnt
->mnt_dir
= mnt_dir_save
;
95 #endif /* NEED_AUTOFS_SPACE_HACK */
96 error
= UNMOUNT_TRAP(mp_save
->mnt
);
98 switch ((error
= errno
)) {
101 plog(XLOG_WARNING
, "unmount: %s is not mounted", mp_save
->mnt
->mnt_dir
);
102 error
= 0; /* Not really an error */
107 * This could happen if the kernel insists on following symlinks
108 * when we try to unmount a direct mountpoint. We need to propagate
109 * the error up so that the top layers know it failed and don't
110 * try to rmdir() the mountpoint or other silly things.
112 plog(XLOG_ERROR
, "mount point %s: %m", mp_save
->mnt
->mnt_dir
);
115 #if defined(MNT2_GEN_OPT_FORCE) && defined(HAVE_UMOUNT2)
119 /* caller determines if forced unmounts should be used */
120 if (unmount_flags
& AMU_UMOUNT_FORCE
) {
121 error
= umount2_fs(mntdir
, unmount_flags
);
125 break; /* all is OK */
128 #endif /* MNT2_GEN_OPT_FORCE && HAVE_UMOUNT2 */
130 dlog("%s: unmount: %m", mp_save
->mnt
->mnt_dir
);
134 dlog("Finished unmount(%s)", mp_save
->mnt
->mnt_dir
);
137 #ifdef MOUNT_TABLE_ON_FILE
139 mp
= mlist
= read_mtab(mntdir
, mnttabname
);
142 * Search the mount table looking for
143 * the correct (ie last) matching entry
147 if (STREQ(mp
->mnt
->mnt_dir
, mntdir
))
153 mnt_free(mp_save
->mnt
);
155 rewrite_mtab(mlist
, mnttabname
);
157 #endif /* MOUNT_TABLE_ON_FILE */
162 plog(XLOG_ERROR
, "Couldn't find how to unmount %s", mntdir
);
164 * Assume it is already unmounted
167 } /* end of "if (mp_save)" statement */
175 #if defined(MNT2_GEN_OPT_FORCE) && defined(HAVE_UMOUNT2)
176 /* force unmount, no questions asked, without touching mnttab file */
178 umount2_fs(const char *mntdir
, u_int unmount_flags
)
182 if (unmount_flags
& AMU_UMOUNT_FORCE
) {
183 plog(XLOG_INFO
, "umount2_fs: trying unmount/forced on %s", mntdir
);
184 error
= umount2(mntdir
, MNT2_GEN_OPT_FORCE
); /* Solaris */
185 if (error
< 0 && (errno
== EINVAL
|| errno
== ENOENT
))
186 error
= 0; /* ignore EINVAL/ENOENT */
188 plog(XLOG_WARNING
, "%s: unmount/force: %m", mntdir
);
190 dlog("%s: unmount/force: OK", mntdir
);
194 #endif /* MNT2_GEN_OPT_FORCE && HAVE_UMOUNT2 */