2006-01-16 19:50:04 +03:00
/*
* Copyright ( C ) Sistina Software , Inc . 1997 - 2003 All rights reserved .
2006-05-18 23:09:15 +04:00
* Copyright ( C ) 2004 - 2006 Red Hat , Inc . All rights reserved .
2006-01-16 19:50:04 +03:00
*
* This copyrighted material is made available to anyone wishing to use ,
* modify , copy , or redistribute it subject to the terms and conditions
2006-09-01 19:05:15 +04:00
* of the GNU General Public License version 2.
2006-01-16 19:50:04 +03:00
*/
# include <linux/slab.h>
# include <linux/spinlock.h>
# include <linux/completion.h>
# include <linux/buffer_head.h>
# include <linux/module.h>
# include <linux/init.h>
2006-02-28 01:23:27 +03:00
# include <linux/gfs2_ondisk.h>
2006-08-30 18:36:52 +04:00
# include <asm/atomic.h>
2009-05-19 13:01:18 +04:00
# include <linux/slow-work.h>
2006-01-16 19:50:04 +03:00
# include "gfs2.h"
2006-02-28 01:23:27 +03:00
# include "incore.h"
2008-10-14 19:05:55 +04:00
# include "super.h"
2006-01-16 19:50:04 +03:00
# include "sys.h"
2006-02-28 01:23:27 +03:00
# include "util.h"
2006-09-07 22:40:21 +04:00
# include "glock.h"
2009-01-08 01:03:37 +03:00
# include "quota.h"
static struct shrinker qd_shrinker = {
. shrink = gfs2_shrink_qd_memory ,
. seeks = DEFAULT_SEEKS ,
} ;
2006-01-16 19:50:04 +03:00
2008-07-26 06:45:34 +04:00
static void gfs2_init_inode_once ( void * foo )
2006-05-19 00:25:27 +04:00
{
struct gfs2_inode * ip = foo ;
2007-05-17 09:10:57 +04:00
inode_init_once ( & ip - > i_inode ) ;
init_rwsem ( & ip - > i_rw_mutex ) ;
2008-11-18 16:38:48 +03:00
INIT_LIST_HEAD ( & ip - > i_trunc_list ) ;
2008-01-10 18:18:55 +03:00
ip - > i_alloc = NULL ;
2006-05-19 00:25:27 +04:00
}
2008-07-26 06:45:34 +04:00
static void gfs2_init_glock_once ( void * foo )
2006-08-30 18:36:52 +04:00
{
struct gfs2_glock * gl = foo ;
2007-05-17 09:10:57 +04:00
INIT_HLIST_NODE ( & gl - > gl_list ) ;
spin_lock_init ( & gl - > gl_spin ) ;
INIT_LIST_HEAD ( & gl - > gl_holders ) ;
2008-11-20 16:39:47 +03:00
INIT_LIST_HEAD ( & gl - > gl_lru ) ;
2007-05-17 09:10:57 +04:00
INIT_LIST_HEAD ( & gl - > gl_ail_list ) ;
atomic_set ( & gl - > gl_ail_count , 0 ) ;
2006-08-30 18:36:52 +04:00
}
2009-12-08 15:12:13 +03:00
static void gfs2_init_gl_aspace_once ( void * foo )
{
struct gfs2_glock * gl = foo ;
struct address_space * mapping = ( struct address_space * ) ( gl + 1 ) ;
gfs2_init_glock_once ( gl ) ;
memset ( mapping , 0 , sizeof ( * mapping ) ) ;
INIT_RADIX_TREE ( & mapping - > page_tree , GFP_ATOMIC ) ;
spin_lock_init ( & mapping - > tree_lock ) ;
spin_lock_init ( & mapping - > i_mmap_lock ) ;
INIT_LIST_HEAD ( & mapping - > private_list ) ;
spin_lock_init ( & mapping - > private_lock ) ;
INIT_RAW_PRIO_TREE_ROOT ( & mapping - > i_mmap ) ;
INIT_LIST_HEAD ( & mapping - > i_mmap_nonlinear ) ;
}
2006-01-16 19:50:04 +03:00
/**
* init_gfs2_fs - Register GFS2 as a filesystem
*
* Returns : 0 on success , error code on failure
*/
static int __init init_gfs2_fs ( void )
{
int error ;
error = gfs2_sys_init ( ) ;
if ( error )
return error ;
2006-09-07 22:40:21 +04:00
error = gfs2_glock_init ( ) ;
if ( error )
goto fail ;
2006-01-16 19:50:04 +03:00
2006-09-07 22:40:21 +04:00
error = - ENOMEM ;
2006-01-16 19:50:04 +03:00
gfs2_glock_cachep = kmem_cache_create ( " gfs2_glock " ,
sizeof ( struct gfs2_glock ) ,
2006-09-25 17:26:04 +04:00
0 , 0 ,
2007-07-20 05:11:58 +04:00
gfs2_init_glock_once ) ;
2006-01-16 19:50:04 +03:00
if ( ! gfs2_glock_cachep )
goto fail ;
2009-12-08 15:12:13 +03:00
gfs2_glock_aspace_cachep = kmem_cache_create ( " gfs2_glock (aspace) " ,
sizeof ( struct gfs2_glock ) +
sizeof ( struct address_space ) ,
0 , 0 , gfs2_init_gl_aspace_once ) ;
if ( ! gfs2_glock_aspace_cachep )
goto fail ;
2006-01-16 19:50:04 +03:00
gfs2_inode_cachep = kmem_cache_create ( " gfs2_inode " ,
sizeof ( struct gfs2_inode ) ,
2006-10-28 03:03:48 +04:00
0 , SLAB_RECLAIM_ACCOUNT |
SLAB_MEM_SPREAD ,
2007-07-20 05:11:58 +04:00
gfs2_init_inode_once ) ;
2006-01-16 19:50:04 +03:00
if ( ! gfs2_inode_cachep )
goto fail ;
gfs2_bufdata_cachep = kmem_cache_create ( " gfs2_bufdata " ,
sizeof ( struct gfs2_bufdata ) ,
2007-07-20 05:11:58 +04:00
0 , 0 , NULL ) ;
2006-01-16 19:50:04 +03:00
if ( ! gfs2_bufdata_cachep )
goto fail ;
2008-01-29 02:20:26 +03:00
gfs2_rgrpd_cachep = kmem_cache_create ( " gfs2_rgrpd " ,
sizeof ( struct gfs2_rgrpd ) ,
0 , 0 , NULL ) ;
if ( ! gfs2_rgrpd_cachep )
goto fail ;
2008-11-17 17:25:37 +03:00
gfs2_quotad_cachep = kmem_cache_create ( " gfs2_quotad " ,
sizeof ( struct gfs2_quota_data ) ,
0 , 0 , NULL ) ;
if ( ! gfs2_quotad_cachep )
goto fail ;
2009-01-08 01:03:37 +03:00
register_shrinker ( & qd_shrinker ) ;
2006-01-16 19:50:04 +03:00
error = register_filesystem ( & gfs2_fs_type ) ;
if ( error )
goto fail ;
2006-03-03 00:33:41 +03:00
error = register_filesystem ( & gfs2meta_fs_type ) ;
if ( error )
goto fail_unregister ;
2009-11-19 21:10:23 +03:00
error = slow_work_register_user ( THIS_MODULE ) ;
2009-05-19 13:01:18 +04:00
if ( error )
goto fail_slow ;
2007-03-16 13:26:37 +03:00
gfs2_register_debugfs ( ) ;
2006-01-16 19:50:04 +03:00
printk ( " GFS2 (built %s %s) installed \n " , __DATE__ , __TIME__ ) ;
return 0 ;
2009-05-19 13:01:18 +04:00
fail_slow :
unregister_filesystem ( & gfs2meta_fs_type ) ;
2006-03-03 00:33:41 +03:00
fail_unregister :
unregister_filesystem ( & gfs2_fs_type ) ;
fail :
2009-01-08 01:03:37 +03:00
unregister_shrinker ( & qd_shrinker ) ;
2007-08-01 16:57:10 +04:00
gfs2_glock_exit ( ) ;
2008-11-17 17:25:37 +03:00
if ( gfs2_quotad_cachep )
kmem_cache_destroy ( gfs2_quotad_cachep ) ;
2008-01-29 02:20:26 +03:00
if ( gfs2_rgrpd_cachep )
kmem_cache_destroy ( gfs2_rgrpd_cachep ) ;
2006-01-16 19:50:04 +03:00
if ( gfs2_bufdata_cachep )
kmem_cache_destroy ( gfs2_bufdata_cachep ) ;
if ( gfs2_inode_cachep )
kmem_cache_destroy ( gfs2_inode_cachep ) ;
2009-12-08 15:12:13 +03:00
if ( gfs2_glock_aspace_cachep )
kmem_cache_destroy ( gfs2_glock_aspace_cachep ) ;
2006-01-16 19:50:04 +03:00
if ( gfs2_glock_cachep )
kmem_cache_destroy ( gfs2_glock_cachep ) ;
gfs2_sys_uninit ( ) ;
return error ;
}
/**
* exit_gfs2_fs - Unregister the file system
*
*/
static void __exit exit_gfs2_fs ( void )
{
2009-01-08 01:03:37 +03:00
unregister_shrinker ( & qd_shrinker ) ;
2007-08-01 16:57:10 +04:00
gfs2_glock_exit ( ) ;
2007-03-16 13:26:37 +03:00
gfs2_unregister_debugfs ( ) ;
2006-01-16 19:50:04 +03:00
unregister_filesystem ( & gfs2_fs_type ) ;
2006-03-03 00:33:41 +03:00
unregister_filesystem ( & gfs2meta_fs_type ) ;
2009-11-19 21:10:23 +03:00
slow_work_unregister_user ( THIS_MODULE ) ;
2006-01-16 19:50:04 +03:00
2008-11-17 17:25:37 +03:00
kmem_cache_destroy ( gfs2_quotad_cachep ) ;
2008-01-29 02:20:26 +03:00
kmem_cache_destroy ( gfs2_rgrpd_cachep ) ;
2006-01-16 19:50:04 +03:00
kmem_cache_destroy ( gfs2_bufdata_cachep ) ;
kmem_cache_destroy ( gfs2_inode_cachep ) ;
2009-12-08 15:12:13 +03:00
kmem_cache_destroy ( gfs2_glock_aspace_cachep ) ;
2006-01-16 19:50:04 +03:00
kmem_cache_destroy ( gfs2_glock_cachep ) ;
gfs2_sys_uninit ( ) ;
}
MODULE_DESCRIPTION ( " Global File System " ) ;
MODULE_AUTHOR ( " Red Hat, Inc. " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_init ( init_gfs2_fs ) ;
module_exit ( exit_gfs2_fs ) ;