2005-04-16 15:20:36 -07:00
/* -*- linux-c -*- --------------------------------------------------------- *
*
* linux / fs / autofs / inode . c
*
* Copyright 1997 - 1998 Transmeta Corporation - - All Rights Reserved
*
* This file is part of the Linux kernel and is made available under
* the terms of the GNU General Public License , version 2 , or at your
* option , any later version , incorporated herein by reference .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
# include <linux/kernel.h>
# include <linux/mm.h>
# include <linux/slab.h>
# include <linux/file.h>
# include <linux/parser.h>
# include <linux/bitops.h>
2006-09-24 11:13:19 -04:00
# include <linux/magic.h>
2005-04-16 15:20:36 -07:00
# include "autofs_i.h"
# include <linux/module.h>
2006-10-19 23:28:36 -07:00
void autofs_kill_sb ( struct super_block * sb )
2005-04-16 15:20:36 -07:00
{
struct autofs_sb_info * sbi = autofs_sbi ( sb ) ;
unsigned int n ;
2006-11-14 02:03:29 -08:00
/*
* In the event of a failure in get_sb_nodev the superblock
* info is not present so nothing else has been setup , so
2006-12-06 20:39:38 -08:00
* just call kill_anon_super when we are called from
* deactivate_super .
2006-11-14 02:03:29 -08:00
*/
if ( ! sbi )
2006-12-06 20:39:38 -08:00
goto out_kill_sb ;
2006-11-14 02:03:29 -08:00
2007-05-10 22:23:06 -07:00
if ( ! sbi - > catatonic )
2005-04-16 15:20:36 -07:00
autofs_catatonic_mode ( sbi ) ; /* Free wait queues, close pipe */
2007-05-10 22:23:08 -07:00
put_pid ( sbi - > oz_pgrp ) ;
2005-09-09 13:01:59 -07:00
autofs_hash_nuke ( sbi ) ;
2007-05-10 22:23:06 -07:00
for ( n = 0 ; n < AUTOFS_MAX_SYMLINKS ; n + + ) {
if ( test_bit ( n , sbi - > symlink_bitmap ) )
2005-04-16 15:20:36 -07:00
kfree ( sbi - > symlink [ n ] . data ) ;
}
kfree ( sb - > s_fs_info ) ;
2006-12-06 20:39:38 -08:00
out_kill_sb :
2005-04-16 15:20:36 -07:00
DPRINTK ( ( " autofs: shutting down \n " ) ) ;
2006-10-19 23:28:36 -07:00
kill_anon_super ( sb ) ;
2005-04-16 15:20:36 -07:00
}
2007-02-12 00:55:41 -08:00
static const struct super_operations autofs_sops = {
2005-04-16 15:20:36 -07:00
. statfs = simple_statfs ,
2008-02-08 04:21:38 -08:00
. show_options = generic_show_options ,
2005-04-16 15:20:36 -07:00
} ;
enum { Opt_err , Opt_fd , Opt_uid , Opt_gid , Opt_pgrp , Opt_minproto , Opt_maxproto } ;
2008-10-13 10:46:57 +01:00
static const match_table_t autofs_tokens = {
2005-04-16 15:20:36 -07:00
{ Opt_fd , " fd=%u " } ,
{ Opt_uid , " uid=%u " } ,
{ Opt_gid , " gid=%u " } ,
{ Opt_pgrp , " pgrp=%u " } ,
{ Opt_minproto , " minproto=%u " } ,
{ Opt_maxproto , " maxproto=%u " } ,
{ Opt_err , NULL }
} ;
2007-05-10 22:23:06 -07:00
static int parse_options ( char * options , int * pipefd , uid_t * uid , gid_t * gid ,
pid_t * pgrp , int * minproto , int * maxproto )
2005-04-16 15:20:36 -07:00
{
char * p ;
substring_t args [ MAX_OPT_ARGS ] ;
int option ;
2008-11-14 10:38:45 +11:00
* uid = current_uid ( ) ;
* gid = current_gid ( ) ;
2007-10-18 23:39:46 -07:00
* pgrp = task_pgrp_nr ( current ) ;
2005-04-16 15:20:36 -07:00
* minproto = * maxproto = AUTOFS_PROTO_VERSION ;
* pipefd = - 1 ;
if ( ! options )
return 1 ;
while ( ( p = strsep ( & options , " , " ) ) ! = NULL ) {
int token ;
if ( ! * p )
continue ;
token = match_token ( p , autofs_tokens , args ) ;
switch ( token ) {
case Opt_fd :
if ( match_int ( & args [ 0 ] , & option ) )
return 1 ;
* pipefd = option ;
break ;
case Opt_uid :
if ( match_int ( & args [ 0 ] , & option ) )
return 1 ;
* uid = option ;
break ;
case Opt_gid :
if ( match_int ( & args [ 0 ] , & option ) )
return 1 ;
* gid = option ;
break ;
case Opt_pgrp :
if ( match_int ( & args [ 0 ] , & option ) )
return 1 ;
* pgrp = option ;
break ;
case Opt_minproto :
if ( match_int ( & args [ 0 ] , & option ) )
return 1 ;
* minproto = option ;
break ;
case Opt_maxproto :
if ( match_int ( & args [ 0 ] , & option ) )
return 1 ;
* maxproto = option ;
break ;
default :
return 1 ;
}
}
return ( * pipefd < 0 ) ;
}
int autofs_fill_super ( struct super_block * s , void * data , int silent )
{
struct inode * root_inode ;
struct dentry * root ;
struct file * pipe ;
int pipefd ;
struct autofs_sb_info * sbi ;
int minproto , maxproto ;
2007-05-10 22:23:08 -07:00
pid_t pgid ;
2005-04-16 15:20:36 -07:00
2008-02-08 04:21:38 -08:00
save_mount_options ( s , data ) ;
2006-09-27 01:49:37 -07:00
sbi = kzalloc ( sizeof ( * sbi ) , GFP_KERNEL ) ;
2007-05-10 22:23:06 -07:00
if ( ! sbi )
2005-04-16 15:20:36 -07:00
goto fail_unlock ;
DPRINTK ( ( " autofs: starting up, sbi = %p \n " , sbi ) ) ;
s - > s_fs_info = sbi ;
sbi - > magic = AUTOFS_SBI_MAGIC ;
2006-11-14 02:03:29 -08:00
sbi - > pipe = NULL ;
sbi - > catatonic = 1 ;
2005-04-16 15:20:36 -07:00
sbi - > exp_timeout = 0 ;
autofs_initialize_hash ( & sbi - > dirhash ) ;
sbi - > queues = NULL ;
memset ( sbi - > symlink_bitmap , 0 , sizeof ( long ) * AUTOFS_SYMLINK_BITMAP_LEN ) ;
sbi - > next_dir_ino = AUTOFS_FIRST_DIR_INO ;
s - > s_blocksize = 1024 ;
s - > s_blocksize_bits = 10 ;
s - > s_magic = AUTOFS_SUPER_MAGIC ;
s - > s_op = & autofs_sops ;
s - > s_time_gran = 1 ;
2005-09-09 13:01:59 -07:00
sbi - > sb = s ;
2005-04-16 15:20:36 -07:00
2008-02-07 00:15:30 -08:00
root_inode = autofs_iget ( s , AUTOFS_ROOT_INO ) ;
if ( IS_ERR ( root_inode ) )
goto fail_free ;
2005-04-16 15:20:36 -07:00
root = d_alloc_root ( root_inode ) ;
pipe = NULL ;
if ( ! root )
goto fail_iput ;
/* Can this call block? - WTF cares? s is locked. */
2007-05-10 22:23:06 -07:00
if ( parse_options ( data , & pipefd , & root_inode - > i_uid ,
2007-05-10 22:23:08 -07:00
& root_inode - > i_gid , & pgid , & minproto ,
2007-05-10 22:23:06 -07:00
& maxproto ) ) {
2005-04-16 15:20:36 -07:00
printk ( " autofs: called with bogus options \n " ) ;
goto fail_dput ;
}
/* Couldn't this be tested earlier? */
2007-05-10 22:23:06 -07:00
if ( minproto > AUTOFS_PROTO_VERSION | |
maxproto < AUTOFS_PROTO_VERSION ) {
2005-04-16 15:20:36 -07:00
printk ( " autofs: kernel does not match daemon version \n " ) ;
goto fail_dput ;
}
2007-05-10 22:23:08 -07:00
DPRINTK ( ( " autofs: pipe fd = %d, pgrp = %u \n " , pipefd , pgid ) ) ;
sbi - > oz_pgrp = find_get_pid ( pgid ) ;
if ( ! sbi - > oz_pgrp ) {
printk ( " autofs: could not find process group %d \n " , pgid ) ;
goto fail_dput ;
}
2005-04-16 15:20:36 -07:00
pipe = fget ( pipefd ) ;
2007-05-10 22:23:06 -07:00
if ( ! pipe ) {
2005-04-16 15:20:36 -07:00
printk ( " autofs: could not open pipe file descriptor \n " ) ;
2007-05-10 22:23:08 -07:00
goto fail_put_pid ;
2005-04-16 15:20:36 -07:00
}
2007-05-10 22:23:08 -07:00
2007-05-10 22:23:06 -07:00
if ( ! pipe - > f_op | | ! pipe - > f_op - > write )
2005-04-16 15:20:36 -07:00
goto fail_fput ;
sbi - > pipe = pipe ;
2006-11-14 02:03:29 -08:00
sbi - > catatonic = 0 ;
2005-04-16 15:20:36 -07:00
/*
* Success ! Install the root dentry now to indicate completion .
*/
s - > s_root = root ;
return 0 ;
fail_fput :
printk ( " autofs: pipe file descriptor does not contain proper ops \n " ) ;
fput ( pipe ) ;
2007-05-10 22:23:08 -07:00
fail_put_pid :
put_pid ( sbi - > oz_pgrp ) ;
2005-04-16 15:20:36 -07:00
fail_dput :
dput ( root ) ;
goto fail_free ;
fail_iput :
printk ( " autofs: get root dentry failed \n " ) ;
iput ( root_inode ) ;
fail_free :
kfree ( sbi ) ;
2006-11-14 02:03:29 -08:00
s - > s_fs_info = NULL ;
2005-04-16 15:20:36 -07:00
fail_unlock :
return - EINVAL ;
}
2008-02-07 00:15:30 -08:00
struct inode * autofs_iget ( struct super_block * sb , unsigned long ino )
2005-04-16 15:20:36 -07:00
{
unsigned int n ;
2008-02-07 00:15:30 -08:00
struct autofs_sb_info * sbi = autofs_sbi ( sb ) ;
struct inode * inode ;
inode = iget_locked ( sb , ino ) ;
if ( ! inode )
return ERR_PTR ( - ENOMEM ) ;
if ( ! ( inode - > i_state & I_NEW ) )
return inode ;
2005-04-16 15:20:36 -07:00
/* Initialize to the default case (stub directory) */
inode - > i_op = & simple_dir_inode_operations ;
inode - > i_fop = & simple_dir_operations ;
inode - > i_mode = S_IFDIR | S_IRUGO | S_IXUGO ;
inode - > i_nlink = 2 ;
inode - > i_mtime = inode - > i_atime = inode - > i_ctime = CURRENT_TIME ;
2007-05-10 22:23:06 -07:00
if ( ino = = AUTOFS_ROOT_INO ) {
2005-04-16 15:20:36 -07:00
inode - > i_mode = S_IFDIR | S_IRUGO | S_IXUGO | S_IWUSR ;
inode - > i_op = & autofs_root_inode_operations ;
inode - > i_fop = & autofs_root_operations ;
2008-02-07 00:15:30 -08:00
goto done ;
2005-04-16 15:20:36 -07:00
}
inode - > i_uid = inode - > i_sb - > s_root - > d_inode - > i_uid ;
inode - > i_gid = inode - > i_sb - > s_root - > d_inode - > i_gid ;
2007-05-10 22:23:06 -07:00
if ( ino > = AUTOFS_FIRST_SYMLINK & & ino < AUTOFS_FIRST_DIR_INO ) {
2005-04-16 15:20:36 -07:00
/* Symlink inode - should be in symlink list */
struct autofs_symlink * sl ;
n = ino - AUTOFS_FIRST_SYMLINK ;
2007-05-10 22:23:06 -07:00
if ( n > = AUTOFS_MAX_SYMLINKS | | ! test_bit ( n , sbi - > symlink_bitmap ) ) {
2005-04-16 15:20:36 -07:00
printk ( " autofs: Looking for bad symlink inode %u \n " , ( unsigned int ) ino ) ;
2008-02-07 00:15:30 -08:00
goto done ;
2005-04-16 15:20:36 -07:00
}
inode - > i_op = & autofs_symlink_inode_operations ;
sl = & sbi - > symlink [ n ] ;
2006-09-27 01:50:46 -07:00
inode - > i_private = sl ;
2005-04-16 15:20:36 -07:00
inode - > i_mode = S_IFLNK | S_IRWXUGO ;
inode - > i_mtime . tv_sec = inode - > i_ctime . tv_sec = sl - > mtime ;
inode - > i_mtime . tv_nsec = inode - > i_ctime . tv_nsec = 0 ;
inode - > i_size = sl - > len ;
inode - > i_nlink = 1 ;
}
2008-02-07 00:15:30 -08:00
done :
unlock_new_inode ( inode ) ;
return inode ;
2005-04-16 15:20:36 -07:00
}