2005-04-09 02:46:31 +04:00
/*
Unmount utility program for Linux CIFS VFS ( virtual filesystem ) client
Copyright ( C ) 2005 Steve French ( sfrench @ us . ibm . com )
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
2007-07-09 23:25:36 +04:00
the Free Software Foundation ; either version 3 of the License , or
2005-04-09 02:46:31 +04:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
2007-07-10 04:52:41 +04:00
along with this program . If not , see < http : //www.gnu.org/licenses/>. */
2005-04-09 02:46:31 +04:00
# ifndef _GNU_SOURCE
# define _GNU_SOURCE
# endif
# include <stdlib.h>
# include <stdio.h>
# include <unistd.h>
# include <ctype.h>
# include <sys/types.h>
# include <sys/mount.h>
# include <sys/ioctl.h>
# include <sys/stat.h>
2005-04-15 09:42:03 +04:00
# include <sys/vfs.h>
2005-04-09 02:46:31 +04:00
# include <fcntl.h>
# include <getopt.h>
# include <errno.h>
# include <string.h>
# include <mntent.h>
2009-02-05 22:12:42 +03:00
# include <limits.h>
2008-11-06 23:07:07 +03:00
# include "mount.h"
2005-04-09 02:46:31 +04:00
# define UNMOUNT_CIFS_VERSION_MAJOR "0"
2008-11-06 23:07:07 +03:00
# define UNMOUNT_CIFS_VERSION_MINOR "6"
2005-04-09 02:46:31 +04:00
# ifndef UNMOUNT_CIFS_VENDOR_SUFFIX
2006-02-27 17:18:52 +03:00
# ifdef _SAMBA_BUILD_
# include "include/version.h"
# ifdef SAMBA_VERSION_VENDOR_SUFFIX
# define UNMOUNT_CIFS_VENDOR_SUFFIX "-"SAMBA_VERSION_OFFICIAL_STRING"-"SAMBA_VERSION_VENDOR_SUFFIX
# else
# define UNMOUNT_CIFS_VENDOR_SUFFIX "-"SAMBA_VERSION_OFFICIAL_STRING
# endif /* SAMBA_VERSION_OFFICIAL_STRING and SAMBA_VERSION_VENDOR_SUFFIX */
# else
# define UNMOUNT_CIFS_VENDOR_SUFFIX ""
# endif /* _SAMBA_BUILD_ */
# endif /* UNMOUNT_CIFS_VENDOR_SUFFIX */
2005-04-09 02:46:31 +04:00
# ifndef MNT_DETACH
# define MNT_DETACH 0x02
# endif
# ifndef MNT_EXPIRE
# define MNT_EXPIRE 0x04
# endif
2005-04-26 09:19:22 +04:00
# ifndef MOUNTED_LOCK
# define MOUNTED_LOCK " / etc / mtab~"
# endif
# ifndef MOUNTED_TEMP
# define MOUNTED_TEMP " / etc / mtab.tmp"
# endif
2005-04-15 09:42:03 +04:00
# define CIFS_IOC_CHECKUMOUNT _IO(0xCF, 2)
# define CIFS_MAGIC_NUMBER 0xFF534D42 /* the first four bytes of SMB PDU */
2005-04-09 02:46:31 +04:00
static struct option longopts [ ] = {
{ " all " , 0 , NULL , ' a ' } ,
{ " help " , 0 , NULL , ' h ' } ,
{ " read-only " , 0 , NULL , ' r ' } ,
{ " ro " , 0 , NULL , ' r ' } ,
{ " verbose " , 0 , NULL , ' v ' } ,
{ " version " , 0 , NULL , ' V ' } ,
{ " expire " , 0 , NULL , ' e ' } ,
{ " force " , 0 , 0 , ' f ' } ,
{ " lazy " , 0 , 0 , ' l ' } ,
{ " no-mtab " , 0 , 0 , ' n ' } ,
{ NULL , 0 , NULL , 0 }
} ;
2006-03-10 20:51:22 +03:00
const char * thisprogram ;
2005-04-09 02:46:31 +04:00
int verboseflg = 0 ;
static void umount_cifs_usage ( void )
{
printf ( " \n Usage: %s <remotetarget> <dir> \n " , thisprogram ) ;
printf ( " \n Unmount the specified directory \n " ) ;
printf ( " \n Less commonly used options: " ) ;
printf ( " \n \t -r \t If mount fails, retry with readonly remount. " ) ;
printf ( " \n \t -n \t Do not write to mtab. " ) ;
printf ( " \n \t -f \t Attempt a forced unmount, even if the fs is busy. " ) ;
printf ( " \n \t -l \t Attempt lazy unmount, Unmount now, cleanup later. " ) ;
printf ( " \n \t -v \t Enable verbose mode (may be useful for debugging). " ) ;
printf ( " \n \t -h \t Display this help. " ) ;
printf ( " \n \n Options are described in more detail in the manual page " ) ;
printf ( " \n \t man 8 umount.cifs \n " ) ;
printf ( " \n To display the version number of the cifs umount utility: " ) ;
printf ( " \n \t %s -V \n " , thisprogram ) ;
2005-06-03 04:34:13 +04:00
printf ( " \n Invoking the umount utility on cifs mounts, can execute " ) ;
printf ( " /sbin/umount.cifs (if present and umount -i is not specified. \n " ) ;
2005-04-09 02:46:31 +04:00
}
static int umount_check_perm ( char * dir )
{
int fileid ;
int rc ;
2005-04-29 02:28:11 +04:00
/* allow root to unmount, no matter what */
if ( getuid ( ) = = 0 )
return 0 ;
2005-04-09 02:46:31 +04:00
2005-04-29 02:28:11 +04:00
/* presumably can not chdir into the target as we do on mount */
2005-04-09 02:46:31 +04:00
fileid = open ( dir , O_RDONLY | O_DIRECTORY | O_NOFOLLOW , 0 ) ;
2005-04-09 09:01:00 +04:00
if ( fileid = = - 1 ) {
if ( verboseflg )
printf ( " error opening mountpoint %d %s " , errno , strerror ( errno ) ) ;
return errno ;
}
2005-04-09 02:46:31 +04:00
rc = ioctl ( fileid , CIFS_IOC_CHECKUMOUNT , NULL ) ;
if ( verboseflg )
2005-04-09 09:01:00 +04:00
printf ( " ioctl returned %d with errno %d %s \n " , rc , errno , strerror ( errno ) ) ;
2005-04-09 02:46:31 +04:00
2005-06-03 04:34:13 +04:00
if ( rc = = ENOTTY ) {
printf ( " user unmounting via %s is an optional feature of " , thisprogram ) ;
printf ( " the cifs filesystem driver (cifs.ko) " ) ;
printf ( " \n \t and requires cifs.ko version 1.32 or later \n " ) ;
2007-12-13 05:45:13 +03:00
} else if ( rc ! = 0 )
2005-04-15 09:42:03 +04:00
printf ( " user unmount of %s failed with %d %s \n " , dir , errno , strerror ( errno ) ) ;
2005-04-09 02:46:31 +04:00
close ( fileid ) ;
return rc ;
}
2006-12-19 23:16:52 +03:00
static int remove_from_mtab ( char * mountpoint )
2005-04-26 09:19:22 +04:00
{
2005-06-03 04:34:13 +04:00
int rc ;
int num_matches ;
2005-04-26 09:19:22 +04:00
FILE * org_fd ;
FILE * new_fd ;
struct mntent * mount_entry ;
/* Do we need to check if it is a symlink to e.g. /proc/mounts
in which case we probably do not want to update it ? */
/* Do we first need to check if it is writable? */
2008-11-06 23:07:07 +03:00
atexit ( unlock_mtab ) ;
2005-06-03 04:34:13 +04:00
if ( lock_mtab ( ) ) {
printf ( " Mount table locked \n " ) ;
return - EACCES ;
}
2005-04-26 09:19:22 +04:00
if ( verboseflg )
printf ( " attempting to remove from mtab \n " ) ;
org_fd = setmntent ( MOUNTED , " r " ) ;
if ( org_fd = = NULL ) {
printf ( " Can not open %s \n " , MOUNTED ) ;
2005-06-03 04:34:13 +04:00
unlock_mtab ( ) ;
2005-04-26 09:19:22 +04:00
return - EIO ;
}
new_fd = setmntent ( MOUNTED_TEMP , " w " ) ;
if ( new_fd = = NULL ) {
printf ( " Can not open temp file %s " , MOUNTED_TEMP ) ;
endmntent ( org_fd ) ;
2005-06-03 04:34:13 +04:00
unlock_mtab ( ) ;
2005-04-26 09:19:22 +04:00
return - EIO ;
}
/* BB fix so we only remove the last entry that matches BB */
2005-06-03 04:34:13 +04:00
num_matches = 0 ;
while ( ( mount_entry = getmntent ( org_fd ) ) ! = NULL ) {
if ( strcmp ( mount_entry - > mnt_dir , mountpoint ) = = 0 ) {
num_matches + + ;
}
}
if ( verboseflg )
printf ( " %d matching entries in mount table \n " , num_matches ) ;
/* Is there a better way to seek back to the first entry in mtab? */
endmntent ( org_fd ) ;
org_fd = setmntent ( MOUNTED , " r " ) ;
if ( org_fd = = NULL ) {
printf ( " Can not open %s \n " , MOUNTED ) ;
unlock_mtab ( ) ;
return - EIO ;
}
2005-04-26 09:19:22 +04:00
while ( ( mount_entry = getmntent ( org_fd ) ) ! = NULL ) {
if ( strcmp ( mount_entry - > mnt_dir , mountpoint ) ! = 0 ) {
addmntent ( new_fd , mount_entry ) ;
} else {
2005-06-03 04:34:13 +04:00
if ( num_matches ! = 1 ) {
addmntent ( new_fd , mount_entry ) ;
num_matches - - ;
} else if ( verboseflg )
printf ( " entry not copied (ie entry is removed) \n " ) ;
2005-04-26 09:19:22 +04:00
}
}
if ( verboseflg )
printf ( " done updating tmp file \n " ) ;
rc = fchmod ( fileno ( new_fd ) , S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH ) ;
if ( rc < 0 ) {
printf ( " error %s changing mode of %s \n " , strerror ( errno ) ,
MOUNTED_TEMP ) ;
}
endmntent ( new_fd ) ;
rc = rename ( MOUNTED_TEMP , MOUNTED ) ;
if ( rc < 0 ) {
printf ( " failure %s renaming %s to %s \n " , strerror ( errno ) ,
MOUNTED_TEMP , MOUNTED ) ;
2005-06-03 04:34:13 +04:00
unlock_mtab ( ) ;
2005-04-26 09:19:22 +04:00
return - EIO ;
}
2005-06-03 04:34:13 +04:00
unlock_mtab ( ) ;
2005-04-26 09:19:22 +04:00
return rc ;
}
2009-02-05 22:12:42 +03:00
/* Make a canonical pathname from PATH. Returns a freshly malloced string.
It is up the * caller * to ensure that the PATH is sensible . i . e .
canonicalize ( " /dev/fd0/. " ) returns " /dev/fd0 " even though ` ` / dev / fd0 / . ' '
is not a legal pathname for ` ` / dev / fd0 ' ' Anything we cannot parse
we return unmodified . */
static char *
canonicalize ( char * path )
{
char * canonical = malloc ( PATH_MAX + 1 ) ;
if ( ! canonical ) {
fprintf ( stderr , " Error! Not enough memory! \n " ) ;
return NULL ;
}
if ( strlen ( path ) > PATH_MAX ) {
fprintf ( stderr , " Mount point string too long \n " ) ;
return NULL ;
}
if ( path = = NULL )
return NULL ;
if ( realpath ( path , canonical ) )
return canonical ;
strncpy ( canonical , path , PATH_MAX ) ;
canonical [ PATH_MAX ] = ' \0 ' ;
return canonical ;
}
2005-04-09 02:46:31 +04:00
int main ( int argc , char * * argv )
{
int c ;
int rc ;
int flags = 0 ;
int nomtab = 0 ;
int retry_remount = 0 ;
2005-04-15 09:42:03 +04:00
struct statfs statbuf ;
2005-04-09 02:46:31 +04:00
char * mountpoint ;
if ( argc & & argv ) {
thisprogram = argv [ 0 ] ;
} else {
umount_cifs_usage ( ) ;
return - EINVAL ;
}
if ( argc < 2 ) {
umount_cifs_usage ( ) ;
return - EINVAL ;
}
if ( thisprogram = = NULL )
thisprogram = " umount.cifs " ;
/* add sharename in opts string as unc= parm */
while ( ( c = getopt_long ( argc , argv , " afhilnrvV " ,
longopts , NULL ) ) ! = - 1 ) {
switch ( c ) {
/* No code to do the following option yet */
/* case 'a':
+ + umount_all ;
break ; */
case ' ? ' :
case ' h ' : /* help */
umount_cifs_usage ( ) ;
exit ( 1 ) ;
case ' n ' :
+ + nomtab ;
break ;
case ' f ' :
flags | = MNT_FORCE ;
break ;
case ' l ' :
flags | = MNT_DETACH ; /* lazy unmount */
break ;
case ' e ' :
flags | = MNT_EXPIRE ; /* gradually timeout */
break ;
case ' r ' :
+ + retry_remount ;
break ;
case ' v ' :
+ + verboseflg ;
break ;
case ' V ' :
printf ( " umount.cifs version: %s.%s%s \n " ,
UNMOUNT_CIFS_VERSION_MAJOR ,
UNMOUNT_CIFS_VERSION_MINOR ,
UNMOUNT_CIFS_VENDOR_SUFFIX ) ;
exit ( 0 ) ;
default :
printf ( " unknown unmount option %c \n " , c ) ;
umount_cifs_usage ( ) ;
exit ( 1 ) ;
}
}
/* move past the umount options */
argv + = optind ;
argc - = optind ;
2009-02-05 22:12:42 +03:00
mountpoint = canonicalize ( argv [ 0 ] ) ;
2005-04-09 02:46:31 +04:00
if ( ( argc < 1 ) | | ( argv [ 0 ] = = NULL ) ) {
printf ( " \n Missing name of unmount directory \n " ) ;
umount_cifs_usage ( ) ;
return - EINVAL ;
}
if ( verboseflg )
printf ( " optind %d unmount dir %s \n " , optind , mountpoint ) ;
/* check if running effectively root */
2005-04-15 09:42:03 +04:00
if ( geteuid ( ) ! = 0 ) {
2005-04-09 02:46:31 +04:00
printf ( " Trying to unmount when %s not installed suid \n " , thisprogram ) ;
2005-04-15 09:42:03 +04:00
if ( verboseflg )
printf ( " euid = %d \n " , geteuid ( ) ) ;
return - EACCES ;
}
2005-04-09 02:46:31 +04:00
/* fixup path if needed */
2007-11-17 00:26:26 +03:00
/* Trim any trailing slashes */
while ( ( strlen ( mountpoint ) > 1 ) & &
( mountpoint [ strlen ( mountpoint ) - 1 ] = = ' / ' ) )
{
mountpoint [ strlen ( mountpoint ) - 1 ] = ' \0 ' ;
}
2005-04-15 09:42:03 +04:00
/* make sure that this is a cifs filesystem */
rc = statfs ( mountpoint , & statbuf ) ;
if ( rc | | ( statbuf . f_type ! = CIFS_MAGIC_NUMBER ) ) {
2005-06-03 04:34:13 +04:00
printf ( " This utility only unmounts cifs filesystems. \n " ) ;
2005-04-15 09:42:03 +04:00
return - EINVAL ;
}
2005-04-09 02:46:31 +04:00
/* check if our uid was the one who mounted */
rc = umount_check_perm ( mountpoint ) ;
if ( rc ) {
2005-04-15 09:42:03 +04:00
printf ( " Not permitted to unmount \n " ) ;
2005-04-09 02:46:31 +04:00
return rc ;
}
if ( umount2 ( mountpoint , flags ) ) {
/* remember to kill daemon on error */
switch ( errno ) {
case 0 :
2005-04-15 09:42:03 +04:00
printf ( " unmount failed but no error number set \n " ) ;
2005-04-09 02:46:31 +04:00
break ;
default :
2005-04-15 09:42:03 +04:00
printf ( " unmount error %d = %s \n " , errno , strerror ( errno ) ) ;
2005-04-09 02:46:31 +04:00
}
2005-06-03 04:34:13 +04:00
printf ( " Refer to the umount.cifs(8) manual page (man 8 umount.cifs) \n " ) ;
2005-04-09 02:46:31 +04:00
return - 1 ;
} else {
2005-04-26 09:19:22 +04:00
if ( verboseflg )
printf ( " umount2 succeeded \n " ) ;
if ( nomtab = = 0 )
remove_from_mtab ( mountpoint ) ;
2005-04-09 02:46:31 +04:00
}
return 0 ;
}