2011-05-26 20:01:36 +04:00
/*
* Cleancache frontend
*
* This code provides the generic " frontend " layer to call a matching
* " backend " driver implementation of cleancache . See
* Documentation / vm / cleancache . txt for more information .
*
* Copyright ( C ) 2009 - 2010 Oracle Corp . All rights reserved .
* Author : Dan Magenheimer
*
* This work is licensed under the terms of the GNU GPL , version 2.
*/
# include <linux/module.h>
# include <linux/fs.h>
# include <linux/exportfs.h>
# include <linux/mm.h>
2011-09-21 20:28:04 +04:00
# include <linux/debugfs.h>
2011-05-26 20:01:36 +04:00
# include <linux/cleancache.h>
/*
* This global enablement flag may be read thousands of times per second
2011-09-21 19:56:28 +04:00
* by cleancache_get / put / invalidate even on systems where cleancache_ops
2011-05-26 20:01:36 +04:00
* is not claimed ( e . g . cleancache is config ' ed on but remains
* disabled ) , so is preferred to the slower alternative : a function
* call that checks a non - global .
*/
2011-09-21 20:21:20 +04:00
int cleancache_enabled __read_mostly ;
2011-05-26 20:01:36 +04:00
EXPORT_SYMBOL ( cleancache_enabled ) ;
/*
* cleancache_ops is set by cleancache_ops_register to contain the pointers
* to the cleancache " backend " implementation functions .
*/
2011-09-21 20:21:20 +04:00
static struct cleancache_ops cleancache_ops __read_mostly ;
2011-05-26 20:01:36 +04:00
2011-09-21 20:28:04 +04:00
/*
* Counters available via / sys / kernel / debug / frontswap ( if debugfs is
* properly configured . These are for information only so are not protected
* against increment races .
*/
static u64 cleancache_succ_gets ;
static u64 cleancache_failed_gets ;
static u64 cleancache_puts ;
static u64 cleancache_invalidates ;
2011-05-26 20:01:36 +04:00
/*
2013-05-01 02:26:56 +04:00
* When no backend is registered all calls to init_fs and init_shared_fs
* are registered and fake poolids ( FAKE_FS_POOLID_OFFSET or
* FAKE_SHARED_FS_POOLID_OFFSET , plus offset in the respective array
* [ shared_ | ] fs_poolid_map ) are given to the respective super block
* ( sb - > cleancache_poolid ) and no tmem_pools are created . When a backend
* registers with cleancache the previous calls to init_fs and init_shared_fs
* are executed to create tmem_pools and set the respective poolids . While no
* backend is registered all " puts " , " gets " and " flushes " are ignored or failed .
*/
# define MAX_INITIALIZABLE_FS 32
# define FAKE_FS_POOLID_OFFSET 1000
# define FAKE_SHARED_FS_POOLID_OFFSET 2000
# define FS_NO_BACKEND (-1)
# define FS_UNKNOWN (-2)
static int fs_poolid_map [ MAX_INITIALIZABLE_FS ] ;
static int shared_fs_poolid_map [ MAX_INITIALIZABLE_FS ] ;
static char * uuids [ MAX_INITIALIZABLE_FS ] ;
/*
* Mutex for the [ shared_ | ] fs_poolid_map to guard against multiple threads
* invoking umount ( and ending in __cleancache_invalidate_fs ) and also multiple
* threads calling mount ( and ending up in __cleancache_init_ [ shared | ] fs ) .
*/
static DEFINE_MUTEX ( poolid_mutex ) ;
/*
* When set to false ( default ) all calls to the cleancache functions , except
* the __cleancache_invalidate_fs and __cleancache_init_ [ shared | ] fs are guarded
* by the if ( ! backend_registered ) return . This means multiple threads ( from
* different filesystems ) will be checking backend_registered . The usage of a
* bool instead of a atomic_t or a bool guarded by a spinlock is OK - we are
* OK if the time between the backend ' s have been initialized ( and
* backend_registered has been set to true ) and when the filesystems start
* actually calling the backends . The inverse ( when unloading ) is obviously
* not good - but this shim does not do that ( yet ) .
*/
static bool backend_registered __read_mostly ;
/*
* The backends and filesystems work all asynchronously . This is b / c the
* backends can be built as modules .
* The usual sequence of events is :
* a ) mount / - > __cleancache_init_fs is called . We set the
* [ shared_ | ] fs_poolid_map and uuids for .
*
* b ) . user does I / Os - > we call the rest of __cleancache_ * functions
* which return immediately as backend_registered is false .
*
* c ) . modprobe zcache - > cleancache_register_ops . We init the backend
* and set backend_registered to true , and for any fs_poolid_map
* ( which is set by __cleancache_init_fs ) we initialize the poolid .
*
* d ) . user does I / Os - > now that backend_registered is true all the
* __cleancache_ * functions can call the backend . They all check
* that fs_poolid_map is valid and if so invoke the backend .
*
* e ) . umount / - > __cleancache_invalidate_fs , the fs_poolid_map is
* reset ( which is the second check in the __cleancache_ * ops
* to call the backend ) .
*
* The sequence of event could also be c ) , followed by a ) , and d ) . and e ) . The
* c ) would not happen anymore . There is also the chance of c ) , and one thread
* doing a ) + d ) , and another doing e ) . For that case we depend on the
* filesystem calling __cleancache_invalidate_fs in the proper sequence ( so
* that it handles all I / Os before it invalidates the fs ( which is last part
* of unmounting process ) .
*
* Note : The acute reader will notice that there is no " rmmod zcache " case .
* This is b / c the functionality for that is not yet implemented and when
* done , will require some extra locking not yet devised .
*/
/*
* Register operations for cleancache , returning previous thus allowing
* detection of multiple backends and possible nesting .
2011-05-26 20:01:36 +04:00
*/
struct cleancache_ops cleancache_register_ops ( struct cleancache_ops * ops )
{
struct cleancache_ops old = cleancache_ops ;
2013-05-01 02:26:56 +04:00
int i ;
2011-05-26 20:01:36 +04:00
2013-05-01 02:26:56 +04:00
mutex_lock ( & poolid_mutex ) ;
2011-05-26 20:01:36 +04:00
cleancache_ops = * ops ;
2013-05-01 02:26:56 +04:00
backend_registered = true ;
for ( i = 0 ; i < MAX_INITIALIZABLE_FS ; i + + ) {
if ( fs_poolid_map [ i ] = = FS_NO_BACKEND )
fs_poolid_map [ i ] = ( * cleancache_ops . init_fs ) ( PAGE_SIZE ) ;
if ( shared_fs_poolid_map [ i ] = = FS_NO_BACKEND )
shared_fs_poolid_map [ i ] = ( * cleancache_ops . init_shared_fs )
( uuids [ i ] , PAGE_SIZE ) ;
}
out :
mutex_unlock ( & poolid_mutex ) ;
2011-05-26 20:01:36 +04:00
return old ;
}
EXPORT_SYMBOL ( cleancache_register_ops ) ;
/* Called by a cleancache-enabled filesystem at time of mount */
void __cleancache_init_fs ( struct super_block * sb )
{
2013-05-01 02:26:56 +04:00
int i ;
mutex_lock ( & poolid_mutex ) ;
for ( i = 0 ; i < MAX_INITIALIZABLE_FS ; i + + ) {
if ( fs_poolid_map [ i ] = = FS_UNKNOWN ) {
sb - > cleancache_poolid = i + FAKE_FS_POOLID_OFFSET ;
if ( backend_registered )
fs_poolid_map [ i ] = ( * cleancache_ops . init_fs ) ( PAGE_SIZE ) ;
else
fs_poolid_map [ i ] = FS_NO_BACKEND ;
break ;
}
}
mutex_unlock ( & poolid_mutex ) ;
2011-05-26 20:01:36 +04:00
}
EXPORT_SYMBOL ( __cleancache_init_fs ) ;
/* Called by a cleancache-enabled clustered filesystem at time of mount */
void __cleancache_init_shared_fs ( char * uuid , struct super_block * sb )
{
2013-05-01 02:26:56 +04:00
int i ;
mutex_lock ( & poolid_mutex ) ;
for ( i = 0 ; i < MAX_INITIALIZABLE_FS ; i + + ) {
if ( shared_fs_poolid_map [ i ] = = FS_UNKNOWN ) {
sb - > cleancache_poolid = i + FAKE_SHARED_FS_POOLID_OFFSET ;
uuids [ i ] = uuid ;
if ( backend_registered )
shared_fs_poolid_map [ i ] = ( * cleancache_ops . init_shared_fs )
( uuid , PAGE_SIZE ) ;
else
shared_fs_poolid_map [ i ] = FS_NO_BACKEND ;
break ;
}
}
mutex_unlock ( & poolid_mutex ) ;
2011-05-26 20:01:36 +04:00
}
EXPORT_SYMBOL ( __cleancache_init_shared_fs ) ;
/*
* If the filesystem uses exportable filehandles , use the filehandle as
* the key , else use the inode number .
*/
static int cleancache_get_key ( struct inode * inode ,
struct cleancache_filekey * key )
{
2012-04-02 22:34:06 +04:00
int ( * fhfn ) ( struct inode * , __u32 * fh , int * , struct inode * ) ;
2011-05-26 20:01:36 +04:00
int len = 0 , maxlen = CLEANCACHE_KEY_MAX ;
struct super_block * sb = inode - > i_sb ;
key - > u . ino = inode - > i_ino ;
if ( sb - > s_export_op ! = NULL ) {
fhfn = sb - > s_export_op - > encode_fh ;
if ( fhfn ) {
2012-04-02 22:34:06 +04:00
len = ( * fhfn ) ( inode , & key - > u . fh [ 0 ] , & maxlen , NULL ) ;
2013-02-17 10:48:11 +04:00
if ( len < = FILEID_ROOT | | len = = FILEID_INVALID )
2011-05-26 20:01:36 +04:00
return - 1 ;
if ( maxlen > CLEANCACHE_KEY_MAX )
return - 1 ;
}
}
return 0 ;
}
2013-05-01 02:26:56 +04:00
/*
* Returns a pool_id that is associated with a given fake poolid .
*/
static int get_poolid_from_fake ( int fake_pool_id )
{
if ( fake_pool_id > = FAKE_SHARED_FS_POOLID_OFFSET )
return shared_fs_poolid_map [ fake_pool_id -
FAKE_SHARED_FS_POOLID_OFFSET ] ;
else if ( fake_pool_id > = FAKE_FS_POOLID_OFFSET )
return fs_poolid_map [ fake_pool_id - FAKE_FS_POOLID_OFFSET ] ;
return FS_NO_BACKEND ;
}
2011-05-26 20:01:36 +04:00
/*
* " Get " data from cleancache associated with the poolid / inode / index
* that were specified when the data was put to cleanache and , if
* successful , use it to fill the specified page with data and return 0.
* The pageframe is unchanged and returns - 1 if the get fails .
* Page must be locked by caller .
2013-05-01 02:26:56 +04:00
*
* The function has two checks before any action is taken - whether
* a backend is registered and whether the sb - > cleancache_poolid
* is correct .
2011-05-26 20:01:36 +04:00
*/
int __cleancache_get_page ( struct page * page )
{
int ret = - 1 ;
int pool_id ;
2013-05-01 02:26:56 +04:00
int fake_pool_id ;
2011-05-26 20:01:36 +04:00
struct cleancache_filekey key = { . u . key = { 0 } } ;
2013-05-01 02:26:56 +04:00
if ( ! backend_registered ) {
cleancache_failed_gets + + ;
goto out ;
}
2011-05-26 20:01:36 +04:00
VM_BUG_ON ( ! PageLocked ( page ) ) ;
2013-05-01 02:26:56 +04:00
fake_pool_id = page - > mapping - > host - > i_sb - > cleancache_poolid ;
if ( fake_pool_id < 0 )
2011-05-26 20:01:36 +04:00
goto out ;
2013-05-01 02:26:56 +04:00
pool_id = get_poolid_from_fake ( fake_pool_id ) ;
2011-05-26 20:01:36 +04:00
if ( cleancache_get_key ( page - > mapping - > host , & key ) < 0 )
goto out ;
2013-05-01 02:26:56 +04:00
if ( pool_id > = 0 )
ret = ( * cleancache_ops . get_page ) ( pool_id ,
key , page - > index , page ) ;
2011-05-26 20:01:36 +04:00
if ( ret = = 0 )
cleancache_succ_gets + + ;
else
cleancache_failed_gets + + ;
out :
return ret ;
}
EXPORT_SYMBOL ( __cleancache_get_page ) ;
/*
* " Put " data from a page to cleancache and associate it with the
* ( previously - obtained per - filesystem ) poolid and the page ' s ,
* inode and page index . Page must be locked . Note that a put_page
* always " succeeds " , though a subsequent get_page may succeed or fail .
2013-05-01 02:26:56 +04:00
*
* The function has two checks before any action is taken - whether
* a backend is registered and whether the sb - > cleancache_poolid
* is correct .
2011-05-26 20:01:36 +04:00
*/
void __cleancache_put_page ( struct page * page )
{
int pool_id ;
2013-05-01 02:26:56 +04:00
int fake_pool_id ;
2011-05-26 20:01:36 +04:00
struct cleancache_filekey key = { . u . key = { 0 } } ;
2013-05-01 02:26:56 +04:00
if ( ! backend_registered ) {
cleancache_puts + + ;
return ;
}
2011-05-26 20:01:36 +04:00
VM_BUG_ON ( ! PageLocked ( page ) ) ;
2013-05-01 02:26:56 +04:00
fake_pool_id = page - > mapping - > host - > i_sb - > cleancache_poolid ;
if ( fake_pool_id < 0 )
return ;
pool_id = get_poolid_from_fake ( fake_pool_id ) ;
2011-05-26 20:01:36 +04:00
if ( pool_id > = 0 & &
2013-05-01 02:26:56 +04:00
cleancache_get_key ( page - > mapping - > host , & key ) > = 0 ) {
2011-05-26 20:01:36 +04:00
( * cleancache_ops . put_page ) ( pool_id , key , page - > index , page ) ;
cleancache_puts + + ;
}
}
EXPORT_SYMBOL ( __cleancache_put_page ) ;
/*
2011-09-21 19:56:28 +04:00
* Invalidate any data from cleancache associated with the poolid and the
2011-05-26 20:01:36 +04:00
* page ' s inode and page index so that a subsequent " get " will fail .
2013-05-01 02:26:56 +04:00
*
* The function has two checks before any action is taken - whether
* a backend is registered and whether the sb - > cleancache_poolid
* is correct .
2011-05-26 20:01:36 +04:00
*/
2011-09-21 19:56:28 +04:00
void __cleancache_invalidate_page ( struct address_space * mapping ,
struct page * page )
2011-05-26 20:01:36 +04:00
{
/* careful... page->mapping is NULL sometimes when this is called */
2013-05-01 02:26:56 +04:00
int pool_id ;
int fake_pool_id = mapping - > host - > i_sb - > cleancache_poolid ;
2011-05-26 20:01:36 +04:00
struct cleancache_filekey key = { . u . key = { 0 } } ;
2013-05-01 02:26:56 +04:00
if ( ! backend_registered )
return ;
if ( fake_pool_id > = 0 ) {
pool_id = get_poolid_from_fake ( fake_pool_id ) ;
if ( pool_id < 0 )
return ;
2011-05-26 20:01:36 +04:00
VM_BUG_ON ( ! PageLocked ( page ) ) ;
if ( cleancache_get_key ( mapping - > host , & key ) > = 0 ) {
2012-01-12 23:03:25 +04:00
( * cleancache_ops . invalidate_page ) ( pool_id ,
2013-05-01 02:26:56 +04:00
key , page - > index ) ;
2011-09-21 20:28:04 +04:00
cleancache_invalidates + + ;
2011-05-26 20:01:36 +04:00
}
}
}
2011-09-21 19:56:28 +04:00
EXPORT_SYMBOL ( __cleancache_invalidate_page ) ;
2011-05-26 20:01:36 +04:00
/*
2011-09-21 19:56:28 +04:00
* Invalidate all data from cleancache associated with the poolid and the
2011-05-26 20:01:36 +04:00
* mappings ' s inode so that all subsequent gets to this poolid / inode
* will fail .
2013-05-01 02:26:56 +04:00
*
* The function has two checks before any action is taken - whether
* a backend is registered and whether the sb - > cleancache_poolid
* is correct .
2011-05-26 20:01:36 +04:00
*/
2011-09-21 19:56:28 +04:00
void __cleancache_invalidate_inode ( struct address_space * mapping )
2011-05-26 20:01:36 +04:00
{
2013-05-01 02:26:56 +04:00
int pool_id ;
int fake_pool_id = mapping - > host - > i_sb - > cleancache_poolid ;
2011-05-26 20:01:36 +04:00
struct cleancache_filekey key = { . u . key = { 0 } } ;
2013-05-01 02:26:56 +04:00
if ( ! backend_registered )
return ;
if ( fake_pool_id < 0 )
return ;
pool_id = get_poolid_from_fake ( fake_pool_id ) ;
2011-05-26 20:01:36 +04:00
if ( pool_id > = 0 & & cleancache_get_key ( mapping - > host , & key ) > = 0 )
2012-01-12 23:03:25 +04:00
( * cleancache_ops . invalidate_inode ) ( pool_id , key ) ;
2011-05-26 20:01:36 +04:00
}
2011-09-21 19:56:28 +04:00
EXPORT_SYMBOL ( __cleancache_invalidate_inode ) ;
2011-05-26 20:01:36 +04:00
/*
* Called by any cleancache - enabled filesystem at time of unmount ;
2013-05-01 02:26:56 +04:00
* note that pool_id is surrendered and may be returned by a subsequent
* cleancache_init_fs or cleancache_init_shared_fs .
2011-05-26 20:01:36 +04:00
*/
2011-09-21 19:56:28 +04:00
void __cleancache_invalidate_fs ( struct super_block * sb )
2011-05-26 20:01:36 +04:00
{
2013-05-01 02:26:56 +04:00
int index ;
int fake_pool_id = sb - > cleancache_poolid ;
int old_poolid = fake_pool_id ;
mutex_lock ( & poolid_mutex ) ;
if ( fake_pool_id > = FAKE_SHARED_FS_POOLID_OFFSET ) {
index = fake_pool_id - FAKE_SHARED_FS_POOLID_OFFSET ;
old_poolid = shared_fs_poolid_map [ index ] ;
shared_fs_poolid_map [ index ] = FS_UNKNOWN ;
uuids [ index ] = NULL ;
} else if ( fake_pool_id > = FAKE_FS_POOLID_OFFSET ) {
index = fake_pool_id - FAKE_FS_POOLID_OFFSET ;
old_poolid = fs_poolid_map [ index ] ;
fs_poolid_map [ index ] = FS_UNKNOWN ;
2011-05-26 20:01:36 +04:00
}
2013-05-01 02:26:56 +04:00
sb - > cleancache_poolid = - 1 ;
if ( backend_registered )
( * cleancache_ops . invalidate_fs ) ( old_poolid ) ;
mutex_unlock ( & poolid_mutex ) ;
2011-05-26 20:01:36 +04:00
}
2011-09-21 19:56:28 +04:00
EXPORT_SYMBOL ( __cleancache_invalidate_fs ) ;
2011-05-26 20:01:36 +04:00
static int __init init_cleancache ( void )
{
2013-05-01 02:26:56 +04:00
int i ;
2011-09-21 20:28:04 +04:00
# ifdef CONFIG_DEBUG_FS
struct dentry * root = debugfs_create_dir ( " cleancache " , NULL ) ;
if ( root = = NULL )
return - ENXIO ;
debugfs_create_u64 ( " succ_gets " , S_IRUGO , root , & cleancache_succ_gets ) ;
debugfs_create_u64 ( " failed_gets " , S_IRUGO ,
root , & cleancache_failed_gets ) ;
debugfs_create_u64 ( " puts " , S_IRUGO , root , & cleancache_puts ) ;
debugfs_create_u64 ( " invalidates " , S_IRUGO ,
root , & cleancache_invalidates ) ;
# endif
2013-05-01 02:26:56 +04:00
for ( i = 0 ; i < MAX_INITIALIZABLE_FS ; i + + ) {
fs_poolid_map [ i ] = FS_UNKNOWN ;
shared_fs_poolid_map [ i ] = FS_UNKNOWN ;
}
cleancache_enabled = 1 ;
2011-05-26 20:01:36 +04:00
return 0 ;
}
module_init ( init_cleancache )