2014-05-06 19:37:13 +02:00
/*
* Copyright ( C ) 1995 , 1996 , 1997 Olaf Kirch < okir @ monad . swb . de >
*
* This file describes the layout of the file handles as passed
* over the wire .
*/
# ifndef _LINUX_NFSD_NFSFH_H
# define _LINUX_NFSD_NFSFH_H
2015-11-17 06:52:23 -05:00
# include <linux/crc32.h>
2014-05-06 19:37:13 +02:00
# include <linux/sunrpc/svc.h>
# include <uapi/linux/nfsd/nfsfh.h>
static inline __u32 ino_t_to_u32 ( ino_t ino )
{
return ( __u32 ) ino ;
}
static inline ino_t u32_to_ino_t ( __u32 uino )
{
return ( ino_t ) uino ;
}
2009-12-04 19:36:06 -05:00
2014-05-06 19:37:13 +02:00
/*
* This is the internal representation of an NFS handle used in knfsd .
* pre_mtime / post_version will be used to support wcc_attr ' s in NFSv3 .
*/
typedef struct svc_fh {
struct knfsd_fh fh_handle ; /* FH data */
2015-09-17 08:28:38 -04:00
int fh_maxsize ; /* max size for fh_handle */
2014-05-06 19:37:13 +02:00
struct dentry * fh_dentry ; /* validated dentry */
struct svc_export * fh_export ; /* export pointer */
2015-09-17 08:28:39 -04:00
bool fh_locked ; /* inode locked by us */
bool fh_want_write ; /* remount protection taken */
2014-05-06 19:37:13 +02:00
# ifdef CONFIG_NFSD_V3
2015-09-17 08:28:39 -04:00
bool fh_post_saved ; /* post-op attrs saved */
bool fh_pre_saved ; /* pre-op attrs saved */
2014-05-06 19:37:13 +02:00
/* Pre-op attributes saved during fh_lock */
__u64 fh_pre_size ; /* size before operation */
struct timespec fh_pre_mtime ; /* mtime before oper */
struct timespec fh_pre_ctime ; /* ctime before oper */
/*
* pre - op nfsv4 change attr : note must check IS_I_VERSION ( inode )
* to find out if it is valid .
*/
u64 fh_pre_change ;
/* Post-op attributes saved in fh_unlock */
struct kstat fh_post_attr ; /* full attrs after operation */
u64 fh_post_change ; /* nfsv4 change; see above */
# endif /* CONFIG_NFSD_V3 */
2009-12-04 19:36:06 -05:00
2014-05-06 19:37:13 +02:00
} svc_fh ;
2009-12-04 19:36:06 -05:00
enum nfsd_fsid {
FSID_DEV = 0 ,
FSID_NUM ,
FSID_MAJOR_MINOR ,
FSID_ENCODE_DEV ,
FSID_UUID4_INUM ,
FSID_UUID8 ,
FSID_UUID16 ,
FSID_UUID16_INUM ,
} ;
enum fsid_source {
FSIDSOURCE_DEV ,
FSIDSOURCE_FSID ,
FSIDSOURCE_UUID ,
} ;
extern enum fsid_source fsid_source ( struct svc_fh * fhp ) ;
2014-06-17 07:44:14 -04:00
/*
* This might look a little large to " inline " but in all calls except
2009-12-04 19:36:06 -05:00
* one , ' vers ' is constant so moste of the function disappears .
2014-06-17 07:44:14 -04:00
*
* In some cases the values are considered to be host endian and in
* others , net endian . fsidv is always considered to be u32 as the
* callers don ' t know which it will be . So we must use __force to keep
* sparse from complaining . Since these values are opaque to the
* client , that shouldn ' t be a problem .
2009-12-04 19:36:06 -05:00
*/
static inline void mk_fsid ( int vers , u32 * fsidv , dev_t dev , ino_t ino ,
u32 fsid , unsigned char * uuid )
{
u32 * up ;
switch ( vers ) {
case FSID_DEV :
2014-06-17 07:44:14 -04:00
fsidv [ 0 ] = ( __force __u32 ) htonl ( ( MAJOR ( dev ) < < 16 ) |
2009-12-04 19:36:06 -05:00
MINOR ( dev ) ) ;
fsidv [ 1 ] = ino_t_to_u32 ( ino ) ;
break ;
case FSID_NUM :
fsidv [ 0 ] = fsid ;
break ;
case FSID_MAJOR_MINOR :
2014-06-17 07:44:14 -04:00
fsidv [ 0 ] = ( __force __u32 ) htonl ( MAJOR ( dev ) ) ;
fsidv [ 1 ] = ( __force __u32 ) htonl ( MINOR ( dev ) ) ;
2009-12-04 19:36:06 -05:00
fsidv [ 2 ] = ino_t_to_u32 ( ino ) ;
break ;
case FSID_ENCODE_DEV :
fsidv [ 0 ] = new_encode_dev ( dev ) ;
fsidv [ 1 ] = ino_t_to_u32 ( ino ) ;
break ;
case FSID_UUID4_INUM :
/* 4 byte fsid and inode number */
up = ( u32 * ) uuid ;
fsidv [ 0 ] = ino_t_to_u32 ( ino ) ;
fsidv [ 1 ] = up [ 0 ] ^ up [ 1 ] ^ up [ 2 ] ^ up [ 3 ] ;
break ;
case FSID_UUID8 :
/* 8 byte fsid */
up = ( u32 * ) uuid ;
fsidv [ 0 ] = up [ 0 ] ^ up [ 2 ] ;
fsidv [ 1 ] = up [ 1 ] ^ up [ 3 ] ;
break ;
case FSID_UUID16 :
/* 16 byte fsid - NFSv3+ only */
memcpy ( fsidv , uuid , 16 ) ;
break ;
case FSID_UUID16_INUM :
/* 8 byte inode and 16 byte fsid */
* ( u64 * ) fsidv = ( u64 ) ino ;
memcpy ( fsidv + 2 , uuid , 16 ) ;
break ;
default : BUG ( ) ;
}
}
static inline int key_len ( int type )
{
switch ( type ) {
case FSID_DEV : return 8 ;
case FSID_NUM : return 4 ;
case FSID_MAJOR_MINOR : return 12 ;
case FSID_ENCODE_DEV : return 8 ;
case FSID_UUID4_INUM : return 8 ;
case FSID_UUID8 : return 8 ;
case FSID_UUID16 : return 16 ;
case FSID_UUID16_INUM : return 24 ;
default : return 0 ;
}
}
/*
* Shorthand for dprintk ( ) ' s
*/
extern char * SVCFH_fmt ( struct svc_fh * fhp ) ;
/*
* Function prototypes
*/
2011-07-26 03:30:54 -04:00
__be32 fh_verify ( struct svc_rqst * , struct svc_fh * , umode_t , int ) ;
2009-12-04 19:36:06 -05:00
__be32 fh_compose ( struct svc_fh * , struct svc_export * , struct dentry * , struct svc_fh * ) ;
__be32 fh_update ( struct svc_fh * ) ;
void fh_put ( struct svc_fh * ) ;
static __inline__ struct svc_fh *
fh_copy ( struct svc_fh * dst , struct svc_fh * src )
{
WARN_ON ( src - > fh_dentry | | src - > fh_locked ) ;
* dst = * src ;
return dst ;
}
static inline void
fh_copy_shallow ( struct knfsd_fh * dst , struct knfsd_fh * src )
{
dst - > fh_size = src - > fh_size ;
memcpy ( & dst - > fh_base , & src - > fh_base , src - > fh_size ) ;
}
static __inline__ struct svc_fh *
fh_init ( struct svc_fh * fhp , int maxsize )
{
memset ( fhp , 0 , sizeof ( * fhp ) ) ;
fhp - > fh_maxsize = maxsize ;
return fhp ;
}
2014-08-14 08:41:48 +02:00
static inline bool fh_match ( struct knfsd_fh * fh1 , struct knfsd_fh * fh2 )
{
if ( fh1 - > fh_size ! = fh2 - > fh_size )
return false ;
if ( memcmp ( fh1 - > fh_base . fh_pad , fh2 - > fh_base . fh_pad , fh1 - > fh_size ) ! = 0 )
return false ;
return true ;
}
2014-08-13 20:56:13 +02:00
static inline bool fh_fsid_match ( struct knfsd_fh * fh1 , struct knfsd_fh * fh2 )
{
if ( fh1 - > fh_fsid_type ! = fh2 - > fh_fsid_type )
return false ;
2015-02-11 16:08:32 +03:00
if ( memcmp ( fh1 - > fh_fsid , fh2 - > fh_fsid , key_len ( fh1 - > fh_fsid_type ) ) ! = 0 )
2014-08-13 20:56:13 +02:00
return false ;
return true ;
}
2015-11-17 06:52:23 -05:00
# ifdef CONFIG_CRC32
/**
* knfsd_fh_hash - calculate the crc32 hash for the filehandle
* @ fh - pointer to filehandle
*
* returns a crc32 hash for the filehandle that is compatible with
* the one displayed by " wireshark " .
*/
static inline u32
knfsd_fh_hash ( struct knfsd_fh * fh )
{
return ~ crc32_le ( 0xFFFFFFFF , ( unsigned char * ) & fh - > fh_base , fh - > fh_size ) ;
}
# else
static inline u32
knfsd_fh_hash ( struct knfsd_fh * fh )
{
return 0 ;
}
# endif
2009-12-04 19:36:06 -05:00
# ifdef CONFIG_NFSD_V3
2014-03-29 10:23:47 +08:00
/*
* The wcc data stored in current_fh should be cleared
* between compound ops .
*/
static inline void
fh_clear_wcc ( struct svc_fh * fhp )
{
2015-09-17 08:28:39 -04:00
fhp - > fh_post_saved = false ;
fhp - > fh_pre_saved = false ;
2014-03-29 10:23:47 +08:00
}
2017-05-11 14:45:06 -04:00
/*
* We could use i_version alone as the change attribute . However ,
* i_version can go backwards after a reboot . On its own that doesn ' t
* necessarily cause a problem , but if i_version goes backwards and then
* is incremented again it could reuse a value that was previously used
* before boot , and a client who queried the two values might
* incorrectly assume nothing changed .
*
* By using both ctime and the i_version counter we guarantee that as
* long as time doesn ' t go backwards we never reuse an old value .
*/
static inline u64 nfsd4_change_attribute ( struct inode * inode )
{
u64 chattr ;
chattr = inode - > i_ctime . tv_sec ;
chattr < < = 30 ;
chattr + = inode - > i_ctime . tv_nsec ;
chattr + = inode - > i_version ;
return chattr ;
}
2009-12-04 19:36:06 -05:00
/*
* Fill in the pre_op attr for the wcc data
*/
static inline void
fill_pre_wcc ( struct svc_fh * fhp )
{
struct inode * inode ;
2015-03-17 22:25:59 +00:00
inode = d_inode ( fhp - > fh_dentry ) ;
2009-12-04 19:36:06 -05:00
if ( ! fhp - > fh_pre_saved ) {
fhp - > fh_pre_mtime = inode - > i_mtime ;
fhp - > fh_pre_ctime = inode - > i_ctime ;
fhp - > fh_pre_size = inode - > i_size ;
2017-05-11 14:45:06 -04:00
fhp - > fh_pre_change = nfsd4_change_attribute ( inode ) ;
2015-09-17 08:28:39 -04:00
fhp - > fh_pre_saved = true ;
2009-12-04 19:36:06 -05:00
}
}
extern void fill_post_wcc ( struct svc_fh * ) ;
# else
2014-03-29 10:23:47 +08:00
# define fh_clear_wcc(ignored)
# define fill_pre_wcc(ignored)
2009-12-04 19:36:06 -05:00
# define fill_post_wcc(notused)
# endif /* CONFIG_NFSD_V3 */
/*
* Lock a file handle / inode
* NOTE : both fh_lock and fh_unlock are done " by hand " in
* vfs . c : nfsd_rename as it needs to grab 2 i_mutex ' s at once
* so , any changes here should be reflected there .
*/
static inline void
fh_lock_nested ( struct svc_fh * fhp , unsigned int subclass )
{
struct dentry * dentry = fhp - > fh_dentry ;
struct inode * inode ;
BUG_ON ( ! dentry ) ;
if ( fhp - > fh_locked ) {
2013-09-16 10:57:01 -04:00
printk ( KERN_WARNING " fh_lock: %pd2 already locked! \n " ,
dentry ) ;
2009-12-04 19:36:06 -05:00
return ;
}
2015-03-17 22:25:59 +00:00
inode = d_inode ( dentry ) ;
2016-01-22 15:40:57 -05:00
inode_lock_nested ( inode , subclass ) ;
2009-12-04 19:36:06 -05:00
fill_pre_wcc ( fhp ) ;
2015-09-17 08:28:39 -04:00
fhp - > fh_locked = true ;
2009-12-04 19:36:06 -05:00
}
static inline void
fh_lock ( struct svc_fh * fhp )
{
fh_lock_nested ( fhp , I_MUTEX_NORMAL ) ;
}
/*
* Unlock a file handle / inode
*/
static inline void
fh_unlock ( struct svc_fh * fhp )
{
if ( fhp - > fh_locked ) {
fill_post_wcc ( fhp ) ;
2016-01-22 15:40:57 -05:00
inode_unlock ( d_inode ( fhp - > fh_dentry ) ) ;
2015-09-17 08:28:39 -04:00
fhp - > fh_locked = false ;
2009-12-04 19:36:06 -05:00
}
}
2014-05-06 19:37:13 +02:00
# endif /* _LINUX_NFSD_NFSFH_H */