2005-04-17 02:20:36 +04:00
/*
* linux / fs / nfs / symlink . c
*
* Copyright ( C ) 1992 Rick Sladkey
*
* Optimization changes Copyright ( C ) 1994 Florian La Roche
*
* Jun 7 1999 , cache symlink lookups in the page cache . - DaveM
*
* nfs symlink handling code
*/
# include <linux/time.h>
# include <linux/errno.h>
# include <linux/sunrpc/clnt.h>
# include <linux/nfs.h>
# include <linux/nfs2.h>
# include <linux/nfs_fs.h>
# include <linux/pagemap.h>
# include <linux/stat.h>
# include <linux/mm.h>
# include <linux/string.h>
/* Symlink caching in the page cache is even more simplistic
* and straight - forward than readdir caching .
*/
static int nfs_symlink_filler ( struct inode * inode , struct page * page )
{
int error ;
2005-08-20 05:02:56 +04:00
error = NFS_PROTO ( inode ) - > readlink ( inode , page , 0 , PAGE_SIZE ) ;
2005-04-17 02:20:36 +04:00
if ( error < 0 )
goto error ;
SetPageUptodate ( page ) ;
unlock_page ( page ) ;
return 0 ;
error :
SetPageError ( page ) ;
unlock_page ( page ) ;
return - EIO ;
}
2015-11-17 18:20:54 +03:00
static const char * nfs_get_link ( struct dentry * dentry ,
2015-12-29 23:58:39 +03:00
struct inode * inode ,
struct delayed_call * done )
2005-04-17 02:20:36 +04:00
{
struct page * page ;
2007-01-24 22:54:55 +03:00
void * err ;
2015-11-18 05:14:24 +03:00
if ( ! dentry ) {
err = ERR_PTR ( nfs_revalidate_mapping_rcu ( inode ) ) ;
if ( err )
return err ;
page = find_get_page ( inode - > i_mapping , 0 ) ;
if ( ! page )
return ERR_PTR ( - ECHILD ) ;
if ( ! PageUptodate ( page ) ) {
put_page ( page ) ;
return ERR_PTR ( - ECHILD ) ;
}
} else {
err = ERR_PTR ( nfs_revalidate_mapping ( inode , inode - > i_mapping ) ) ;
if ( err )
return err ;
page = read_cache_page ( & inode - > i_data , 0 ,
( filler_t * ) nfs_symlink_filler , inode ) ;
if ( IS_ERR ( page ) )
return ERR_CAST ( page ) ;
}
2015-12-29 23:58:39 +03:00
set_delayed_call ( done , page_put_link , page ) ;
2015-11-17 09:07:57 +03:00
return page_address ( page ) ;
2005-04-17 02:20:36 +04:00
}
/*
* symlinks can ' t do much . . .
*/
2007-02-12 11:55:39 +03:00
const struct inode_operations nfs_symlink_inode_operations = {
2005-04-17 02:20:36 +04:00
. readlink = generic_readlink ,
2015-11-17 18:20:54 +03:00
. get_link = nfs_get_link ,
2005-04-17 02:20:36 +04:00
. getattr = nfs_getattr ,
. setattr = nfs_setattr ,
} ;