1998-01-05 22:56:29 +00:00
/*
* smbumount . c
*
1998-01-22 13:27:43 +00:00
* Copyright ( C ) 1995 - 1998 by Volker Lendecke
1998-01-05 22:56:29 +00:00
*
*/
1998-09-28 19:18:21 +00:00
# include "includes.h"
1998-01-05 22:56:29 +00:00
# include <mntent.h>
1998-09-28 19:18:21 +00:00
# include <asm/types.h>
# include <asm/posix_types.h>
1998-01-05 22:56:29 +00:00
# include <linux/smb.h>
# include <linux/smb_mount.h>
# include <linux/smb_fs.h>
1999-01-31 21:28:55 +00:00
/* This is a (hopefully) temporary hack due to the fact that
sizeof ( uid_t ) ! = sizeof ( __kernel_uid_t ) under glibc .
This may change in the future and smb . h may get fixed in the
future . In the mean time , it ' s ugly hack time - get over it .
*/
# undef SMB_IOC_GETMOUNTUID
# define SMB_IOC_GETMOUNTUID _IOR('u', 1, __kernel_uid_t)
1999-12-13 13:27:58 +00:00
# ifndef O_NOFOLLOW
# define O_NOFOLLOW 0400000
# endif
1998-01-05 22:56:29 +00:00
static void
usage ( void )
{
1999-12-13 13:27:58 +00:00
printf ( " usage: smbumount mountpoint \n " ) ;
1998-01-05 22:56:29 +00:00
}
static int
umount_ok ( const char * mount_point )
{
1999-12-13 13:27:58 +00:00
/* we set O_NOFOLLOW to prevent users playing games with symlinks to
umount filesystems they don ' t own */
int fid = open ( mount_point , O_RDONLY | O_NOFOLLOW , 0 ) ;
1999-02-13 18:36:49 +00:00
__kernel_uid_t mount_uid ;
1999-12-13 13:27:58 +00:00
1998-01-05 22:56:29 +00:00
if ( fid = = - 1 ) {
fprintf ( stderr , " Could not open %s: %s \n " ,
mount_point , strerror ( errno ) ) ;
return - 1 ;
}
if ( ioctl ( fid , SMB_IOC_GETMOUNTUID , & mount_uid ) ! = 0 ) {
fprintf ( stderr , " %s probably not smb-filesystem \n " ,
mount_point ) ;
return - 1 ;
}
1999-12-13 13:27:58 +00:00
if ( ( getuid ( ) ! = 0 )
1998-01-05 22:56:29 +00:00
& & ( mount_uid ! = getuid ( ) ) ) {
fprintf ( stderr , " You are not allowed to umount %s \n " ,
mount_point ) ;
return - 1 ;
}
close ( fid ) ;
return 0 ;
}
/* 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 / . ' '
1999-12-13 13:27:58 +00:00
is not a legal pathname for ` ` / dev / fd0 ' ' Anything we cannot parse
1998-01-05 22:56:29 +00:00
we return unmodified . */
1999-12-13 13:27:58 +00:00
static char *
1998-09-30 03:02:35 +00:00
canonicalize ( char * path )
1998-01-05 22:56:29 +00:00
{
char * canonical = malloc ( PATH_MAX + 1 ) ;
2001-08-08 16:54:16 +00:00
if ( ! canonical ) {
fprintf ( stderr , " Error! Not enough memory! \n " ) ;
return NULL ;
}
1999-12-13 13:27:58 +00:00
if ( strlen ( path ) > PATH_MAX ) {
1998-01-05 22:56:29 +00:00
fprintf ( stderr , " Mount point string too long \n " ) ;
return NULL ;
}
if ( path = = NULL )
return NULL ;
if ( realpath ( path , canonical ) )
return canonical ;
1998-05-12 00:55:32 +00:00
pstrcpy ( canonical , path ) ;
1998-01-05 22:56:29 +00:00
return canonical ;
}
int
main ( int argc , char * argv [ ] )
{
int fd ;
char * mount_point ;
struct mntent * mnt ;
FILE * mtab ;
FILE * new_mtab ;
if ( argc ! = 2 ) {
usage ( ) ;
exit ( 1 ) ;
}
if ( geteuid ( ) ! = 0 ) {
1999-12-13 13:27:58 +00:00
fprintf ( stderr , " smbumount must be installed suid root \n " ) ;
1998-01-05 22:56:29 +00:00
exit ( 1 ) ;
}
mount_point = canonicalize ( argv [ 1 ] ) ;
if ( mount_point = = NULL )
{
exit ( 1 ) ;
}
if ( umount_ok ( mount_point ) ! = 0 ) {
exit ( 1 ) ;
}
if ( umount ( mount_point ) ! = 0 ) {
fprintf ( stderr , " Could not umount %s: %s \n " ,
mount_point , strerror ( errno ) ) ;
exit ( 1 ) ;
}
if ( ( fd = open ( MOUNTED " ~ " , O_RDWR | O_CREAT | O_EXCL , 0600 ) ) = = - 1 )
{
fprintf ( stderr , " Can't get " MOUNTED " ~ lock file " ) ;
return 1 ;
}
close ( fd ) ;
if ( ( mtab = setmntent ( MOUNTED , " r " ) ) = = NULL ) {
fprintf ( stderr , " Can't open " MOUNTED " : %s \n " ,
strerror ( errno ) ) ;
return 1 ;
}
# define MOUNTED_TMP MOUNTED".tmp"
if ( ( new_mtab = setmntent ( MOUNTED_TMP , " w " ) ) = = NULL ) {
fprintf ( stderr , " Can't open " MOUNTED_TMP " : %s \n " ,
strerror ( errno ) ) ;
endmntent ( mtab ) ;
return 1 ;
}
while ( ( mnt = getmntent ( mtab ) ) ! = NULL ) {
if ( strcmp ( mnt - > mnt_dir , mount_point ) ! = 0 ) {
addmntent ( new_mtab , mnt ) ;
}
}
endmntent ( mtab ) ;
if ( fchmod ( fileno ( new_mtab ) , S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH ) < 0 ) {
fprintf ( stderr , " Error changing mode of %s: %s \n " ,
MOUNTED_TMP , strerror ( errno ) ) ;
exit ( 1 ) ;
}
endmntent ( new_mtab ) ;
if ( rename ( MOUNTED_TMP , MOUNTED ) < 0 ) {
fprintf ( stderr , " Cannot rename %s to %s: %s \n " ,
MOUNTED , MOUNTED_TMP , strerror ( errno ) ) ;
exit ( 1 ) ;
}
if ( unlink ( MOUNTED " ~ " ) = = - 1 )
{
fprintf ( stderr , " Can't remove " MOUNTED " ~ " ) ;
return 1 ;
}
return 0 ;
}