2009-04-07 06:01:35 +04:00
/*
* super . c - NILFS module and super block management .
*
* Copyright ( C ) 2005 - 2008 Nippon Telegraph and Telephone Corporation .
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin St , Fifth Floor , Boston , MA 02110 - 1301 USA
*
* Written by Ryusuke Konishi < ryusuke @ osrg . net >
*/
/*
* linux / fs / ext2 / super . c
*
* Copyright ( C ) 1992 , 1993 , 1994 , 1995
* Remy Card ( card @ masi . ibp . fr )
* Laboratoire MASI - Institut Blaise Pascal
* Universite Pierre et Marie Curie ( Paris VI )
*
* from
*
* linux / fs / minix / inode . c
*
* Copyright ( C ) 1991 , 1992 Linus Torvalds
*
* Big - endian to little - endian byte - swapping / bitmaps by
* David S . Miller ( davem @ caip . rutgers . edu ) , 1995
*/
# include <linux/module.h>
# include <linux/string.h>
# include <linux/slab.h>
# include <linux/init.h>
# include <linux/blkdev.h>
# include <linux/parser.h>
# include <linux/random.h>
# include <linux/crc32.h>
# include <linux/vfs.h>
# include <linux/writeback.h>
# include <linux/kobject.h>
2009-06-24 15:06:34 +04:00
# include <linux/seq_file.h>
# include <linux/mount.h>
2009-04-07 06:01:35 +04:00
# include "nilfs.h"
2010-08-26 19:23:02 +04:00
# include "export.h"
2009-04-07 06:01:35 +04:00
# include "mdt.h"
# include "alloc.h"
2010-07-10 15:52:09 +04:00
# include "btree.h"
# include "btnode.h"
2009-04-07 06:01:35 +04:00
# include "page.h"
# include "cpfile.h"
# include "ifile.h"
# include "dat.h"
# include "segment.h"
# include "segbuf.h"
MODULE_AUTHOR ( " NTT Corp. " ) ;
MODULE_DESCRIPTION ( " A New Implementation of the Log-structured Filesystem "
" (NILFS) " ) ;
MODULE_LICENSE ( " GPL " ) ;
2010-10-08 17:37:27 +04:00
static struct kmem_cache * nilfs_inode_cachep ;
2010-04-05 20:54:11 +04:00
struct kmem_cache * nilfs_transaction_cachep ;
struct kmem_cache * nilfs_segbuf_cachep ;
struct kmem_cache * nilfs_btree_path_cache ;
2010-09-20 13:19:06 +04:00
static int nilfs_setup_super ( struct nilfs_sb_info * sbi , int is_mount ) ;
2009-04-07 06:01:35 +04:00
static int nilfs_remount ( struct super_block * sb , int * flags , char * data ) ;
2010-06-28 12:49:30 +04:00
static void nilfs_set_error ( struct nilfs_sb_info * sbi )
{
struct the_nilfs * nilfs = sbi - > s_nilfs ;
2010-06-28 12:49:32 +04:00
struct nilfs_super_block * * sbp ;
2010-06-28 12:49:30 +04:00
down_write ( & nilfs - > ns_sem ) ;
if ( ! ( nilfs - > ns_mount_state & NILFS_ERROR_FS ) ) {
nilfs - > ns_mount_state | = NILFS_ERROR_FS ;
2010-06-28 12:49:33 +04:00
sbp = nilfs_prepare_super ( sbi , 0 ) ;
2010-06-28 12:49:32 +04:00
if ( likely ( sbp ) ) {
sbp [ 0 ] - > s_state | = cpu_to_le16 ( NILFS_ERROR_FS ) ;
2010-06-28 12:49:33 +04:00
if ( sbp [ 1 ] )
sbp [ 1 ] - > s_state | = cpu_to_le16 ( NILFS_ERROR_FS ) ;
nilfs_commit_super ( sbi , NILFS_SB_COMMIT_ALL ) ;
2010-06-28 12:49:32 +04:00
}
2010-06-28 12:49:30 +04:00
}
up_write ( & nilfs - > ns_sem ) ;
}
2009-04-07 06:01:35 +04:00
/**
* nilfs_error ( ) - report failure condition on a filesystem
*
* nilfs_error ( ) sets an ERROR_FS flag on the superblock as well as
* reporting an error message . It should be called when NILFS detects
* incoherences or defects of meta data on disk . As for sustainable
* errors such as a single - shot I / O error , nilfs_warning ( ) or the printk ( )
* function should be used instead .
*
* The segment constructor must not call this function because it can
* kill itself .
*/
void nilfs_error ( struct super_block * sb , const char * function ,
const char * fmt , . . . )
{
struct nilfs_sb_info * sbi = NILFS_SB ( sb ) ;
va_list args ;
va_start ( args , fmt ) ;
printk ( KERN_CRIT " NILFS error (device %s): %s: " , sb - > s_id , function ) ;
vprintk ( fmt , args ) ;
printk ( " \n " ) ;
va_end ( args ) ;
if ( ! ( sb - > s_flags & MS_RDONLY ) ) {
2010-06-28 12:49:30 +04:00
nilfs_set_error ( sbi ) ;
2009-04-07 06:01:35 +04:00
if ( nilfs_test_opt ( sbi , ERRORS_RO ) ) {
printk ( KERN_CRIT " Remounting filesystem read-only \n " ) ;
sb - > s_flags | = MS_RDONLY ;
}
}
if ( nilfs_test_opt ( sbi , ERRORS_PANIC ) )
panic ( " NILFS (device %s): panic forced after error \n " ,
sb - > s_id ) ;
}
void nilfs_warning ( struct super_block * sb , const char * function ,
const char * fmt , . . . )
{
va_list args ;
va_start ( args , fmt ) ;
printk ( KERN_WARNING " NILFS warning (device %s): %s: " ,
sb - > s_id , function ) ;
vprintk ( fmt , args ) ;
printk ( " \n " ) ;
va_end ( args ) ;
}
2010-09-05 08:35:53 +04:00
struct inode * nilfs_alloc_inode ( struct super_block * sb )
2009-04-07 06:01:35 +04:00
{
struct nilfs_inode_info * ii ;
ii = kmem_cache_alloc ( nilfs_inode_cachep , GFP_NOFS ) ;
if ( ! ii )
return NULL ;
ii - > i_bh = NULL ;
ii - > i_state = 0 ;
2010-08-20 16:20:29 +04:00
ii - > i_cno = 0 ;
2009-04-07 06:01:35 +04:00
ii - > vfs_inode . i_version = 1 ;
2010-09-05 08:35:53 +04:00
nilfs_btnode_cache_init ( & ii - > i_btnode_cache , sb - > s_bdi ) ;
2009-04-07 06:01:35 +04:00
return & ii - > vfs_inode ;
}
void nilfs_destroy_inode ( struct inode * inode )
{
2010-08-20 18:46:06 +04:00
struct nilfs_mdt_info * mdi = NILFS_MDT ( inode ) ;
if ( mdi ) {
kfree ( mdi - > mi_bgl ) ; /* kfree(NULL) is safe */
kfree ( mdi ) ;
}
2009-04-07 06:01:35 +04:00
kmem_cache_free ( nilfs_inode_cachep , NILFS_I ( inode ) ) ;
}
2010-06-28 12:49:33 +04:00
static int nilfs_sync_super ( struct nilfs_sb_info * sbi , int flag )
2009-04-07 06:01:35 +04:00
{
struct the_nilfs * nilfs = sbi - > s_nilfs ;
int err ;
retry :
2009-04-07 06:01:59 +04:00
set_buffer_dirty ( nilfs - > ns_sbh [ 0 ] ) ;
2010-08-11 19:05:45 +04:00
if ( nilfs_test_opt ( sbi , BARRIER ) ) {
err = __sync_dirty_buffer ( nilfs - > ns_sbh [ 0 ] ,
2010-08-18 13:29:15 +04:00
WRITE_SYNC | WRITE_FLUSH_FUA ) ;
2010-08-11 19:05:45 +04:00
} else {
err = sync_dirty_buffer ( nilfs - > ns_sbh [ 0 ] ) ;
2009-04-07 06:01:35 +04:00
}
2010-08-11 19:05:45 +04:00
2009-04-07 06:01:59 +04:00
if ( unlikely ( err ) ) {
2009-04-07 06:01:35 +04:00
printk ( KERN_ERR
" NILFS: unable to write superblock (err=%d) \n " , err ) ;
2009-04-07 06:01:59 +04:00
if ( err = = - EIO & & nilfs - > ns_sbh [ 1 ] ) {
2010-06-28 12:49:33 +04:00
/*
* sbp [ 0 ] points to newer log than sbp [ 1 ] ,
* so copy sbp [ 0 ] to sbp [ 1 ] to take over sbp [ 0 ] .
*/
memcpy ( nilfs - > ns_sbp [ 1 ] , nilfs - > ns_sbp [ 0 ] ,
nilfs - > ns_sbsize ) ;
2009-04-07 06:01:59 +04:00
nilfs_fall_back_super_block ( nilfs ) ;
goto retry ;
}
} else {
struct nilfs_super_block * sbp = nilfs - > ns_sbp [ 0 ] ;
2010-06-28 12:49:33 +04:00
nilfs - > ns_sbwcount + + ;
2009-04-07 06:01:59 +04:00
/*
* The latest segment becomes trailable from the position
* written in superblock .
*/
2009-04-07 06:01:35 +04:00
clear_nilfs_discontinued ( nilfs ) ;
2009-04-07 06:01:59 +04:00
/* update GC protection for recent segments */
if ( nilfs - > ns_sbh [ 1 ] ) {
2010-06-28 12:49:33 +04:00
if ( flag = = NILFS_SB_COMMIT_ALL ) {
2009-04-07 06:01:59 +04:00
set_buffer_dirty ( nilfs - > ns_sbh [ 1 ] ) ;
2010-06-28 12:49:33 +04:00
if ( sync_dirty_buffer ( nilfs - > ns_sbh [ 1 ] ) < 0 )
goto out ;
2009-04-07 06:01:59 +04:00
}
2010-06-28 12:49:33 +04:00
if ( le64_to_cpu ( nilfs - > ns_sbp [ 1 ] - > s_last_cno ) <
le64_to_cpu ( nilfs - > ns_sbp [ 0 ] - > s_last_cno ) )
sbp = nilfs - > ns_sbp [ 1 ] ;
2009-04-07 06:01:59 +04:00
}
2009-04-07 06:01:35 +04:00
2010-06-28 12:49:33 +04:00
spin_lock ( & nilfs - > ns_last_segment_lock ) ;
nilfs - > ns_prot_seq = le64_to_cpu ( sbp - > s_last_seq ) ;
spin_unlock ( & nilfs - > ns_last_segment_lock ) ;
}
out :
2009-04-07 06:01:35 +04:00
return err ;
}
2010-06-28 12:49:31 +04:00
void nilfs_set_log_cursor ( struct nilfs_super_block * sbp ,
struct the_nilfs * nilfs )
{
sector_t nfreeblocks ;
/* nilfs->ns_sem must be locked by the caller. */
nilfs_count_free_blocks ( nilfs , & nfreeblocks ) ;
sbp - > s_free_blocks_count = cpu_to_le64 ( nfreeblocks ) ;
spin_lock ( & nilfs - > ns_last_segment_lock ) ;
sbp - > s_last_seq = cpu_to_le64 ( nilfs - > ns_last_seq ) ;
sbp - > s_last_pseg = cpu_to_le64 ( nilfs - > ns_last_pseg ) ;
sbp - > s_last_cno = cpu_to_le64 ( nilfs - > ns_last_cno ) ;
spin_unlock ( & nilfs - > ns_last_segment_lock ) ;
}
2010-06-28 12:49:33 +04:00
struct nilfs_super_block * * nilfs_prepare_super ( struct nilfs_sb_info * sbi ,
int flip )
2009-04-07 06:01:35 +04:00
{
struct the_nilfs * nilfs = sbi - > s_nilfs ;
2009-04-07 06:01:59 +04:00
struct nilfs_super_block * * sbp = nilfs - > ns_sbp ;
2009-04-07 06:01:35 +04:00
2010-06-28 12:49:32 +04:00
/* nilfs->ns_sem must be locked by the caller. */
2010-05-01 05:07:07 +04:00
if ( sbp [ 0 ] - > s_magic ! = cpu_to_le16 ( NILFS_SUPER_MAGIC ) ) {
2010-06-28 12:49:32 +04:00
if ( sbp [ 1 ] & &
sbp [ 1 ] - > s_magic = = cpu_to_le16 ( NILFS_SUPER_MAGIC ) ) {
2010-06-28 12:49:33 +04:00
memcpy ( sbp [ 0 ] , sbp [ 1 ] , nilfs - > ns_sbsize ) ;
2010-06-28 12:49:32 +04:00
} else {
2009-04-07 06:01:59 +04:00
printk ( KERN_CRIT " NILFS: superblock broke on dev %s \n " ,
sbi - > s_super - > s_id ) ;
2010-06-28 12:49:32 +04:00
return NULL ;
2009-04-07 06:01:59 +04:00
}
2010-06-28 12:49:33 +04:00
} else if ( sbp [ 1 ] & &
sbp [ 1 ] - > s_magic ! = cpu_to_le16 ( NILFS_SUPER_MAGIC ) ) {
memcpy ( sbp [ 1 ] , sbp [ 0 ] , nilfs - > ns_sbsize ) ;
2009-04-07 06:01:59 +04:00
}
2010-06-28 12:49:33 +04:00
if ( flip & & sbp [ 1 ] )
nilfs_swap_super_block ( nilfs ) ;
2010-06-28 12:49:32 +04:00
return sbp ;
}
2010-06-28 12:49:33 +04:00
int nilfs_commit_super ( struct nilfs_sb_info * sbi , int flag )
2010-06-28 12:49:32 +04:00
{
struct the_nilfs * nilfs = sbi - > s_nilfs ;
struct nilfs_super_block * * sbp = nilfs - > ns_sbp ;
time_t t ;
/* nilfs->ns_sem must be locked by the caller. */
2009-04-07 06:01:59 +04:00
t = get_seconds ( ) ;
2010-06-28 12:49:33 +04:00
nilfs - > ns_sbwtime = t ;
2009-04-07 06:01:59 +04:00
sbp [ 0 ] - > s_wtime = cpu_to_le64 ( t ) ;
sbp [ 0 ] - > s_sum = 0 ;
sbp [ 0 ] - > s_sum = cpu_to_le32 ( crc32_le ( nilfs - > ns_crc_seed ,
( unsigned char * ) sbp [ 0 ] ,
nilfs - > ns_sbsize ) ) ;
2010-06-28 12:49:33 +04:00
if ( flag = = NILFS_SB_COMMIT_ALL & & sbp [ 1 ] ) {
sbp [ 1 ] - > s_wtime = sbp [ 0 ] - > s_wtime ;
sbp [ 1 ] - > s_sum = 0 ;
sbp [ 1 ] - > s_sum = cpu_to_le32 ( crc32_le ( nilfs - > ns_crc_seed ,
( unsigned char * ) sbp [ 1 ] ,
nilfs - > ns_sbsize ) ) ;
2009-04-07 06:01:59 +04:00
}
2009-12-08 18:57:52 +03:00
clear_nilfs_sb_dirty ( nilfs ) ;
2010-06-28 12:49:33 +04:00
return nilfs_sync_super ( sbi , flag ) ;
2009-04-07 06:01:35 +04:00
}
2010-06-28 12:49:29 +04:00
/**
* nilfs_cleanup_super ( ) - write filesystem state for cleanup
* @ sbi : nilfs_sb_info to be unmounted or degraded to read - only
*
* This function restores state flags in the on - disk super block .
* This will set " clean " flag ( i . e . NILFS_VALID_FS ) unless the
* filesystem was not clean previously .
*/
int nilfs_cleanup_super ( struct nilfs_sb_info * sbi )
{
2010-06-28 12:49:32 +04:00
struct nilfs_super_block * * sbp ;
2010-06-28 12:49:33 +04:00
int flag = NILFS_SB_COMMIT ;
2010-06-28 12:49:32 +04:00
int ret = - EIO ;
2010-06-28 12:49:29 +04:00
2010-06-28 12:49:33 +04:00
sbp = nilfs_prepare_super ( sbi , 0 ) ;
2010-06-28 12:49:32 +04:00
if ( sbp ) {
sbp [ 0 ] - > s_state = cpu_to_le16 ( sbi - > s_nilfs - > ns_mount_state ) ;
2010-06-28 12:49:33 +04:00
nilfs_set_log_cursor ( sbp [ 0 ] , sbi - > s_nilfs ) ;
if ( sbp [ 1 ] & & sbp [ 0 ] - > s_last_cno = = sbp [ 1 ] - > s_last_cno ) {
/*
* make the " clean " flag also to the opposite
* super block if both super blocks point to
* the same checkpoint .
*/
sbp [ 1 ] - > s_state = sbp [ 0 ] - > s_state ;
flag = NILFS_SB_COMMIT_ALL ;
}
ret = nilfs_commit_super ( sbi , flag ) ;
2010-06-28 12:49:32 +04:00
}
2010-06-28 12:49:29 +04:00
return ret ;
}
2009-04-07 06:01:35 +04:00
static void nilfs_put_super ( struct super_block * sb )
{
struct nilfs_sb_info * sbi = NILFS_SB ( sb ) ;
struct the_nilfs * nilfs = sbi - > s_nilfs ;
nilfs_detach_segment_constructor ( sbi ) ;
if ( ! ( sb - > s_flags & MS_RDONLY ) ) {
down_write ( & nilfs - > ns_sem ) ;
2010-06-28 12:49:29 +04:00
nilfs_cleanup_super ( sbi ) ;
2009-04-07 06:01:35 +04:00
up_write ( & nilfs - > ns_sem ) ;
}
2010-09-05 07:20:59 +04:00
iput ( nilfs - > ns_sufile ) ;
iput ( nilfs - > ns_cpfile ) ;
iput ( nilfs - > ns_dat ) ;
2010-09-08 21:07:56 +04:00
destroy_nilfs ( nilfs ) ;
2009-04-07 06:01:35 +04:00
sbi - > s_super = NULL ;
sb - > s_fs_info = NULL ;
2010-08-15 20:54:52 +04:00
kfree ( sbi ) ;
2009-04-07 06:01:35 +04:00
}
static int nilfs_sync_fs ( struct super_block * sb , int wait )
{
2009-07-22 20:26:33 +04:00
struct nilfs_sb_info * sbi = NILFS_SB ( sb ) ;
struct the_nilfs * nilfs = sbi - > s_nilfs ;
2010-06-28 12:49:32 +04:00
struct nilfs_super_block * * sbp ;
2009-04-07 06:01:35 +04:00
int err = 0 ;
/* This function is called when super block should be written back */
if ( wait )
err = nilfs_construct_segment ( sb ) ;
2009-07-22 20:26:33 +04:00
down_write ( & nilfs - > ns_sem ) ;
2010-06-28 12:49:32 +04:00
if ( nilfs_sb_dirty ( nilfs ) ) {
2010-06-28 12:49:33 +04:00
sbp = nilfs_prepare_super ( sbi , nilfs_sb_will_flip ( nilfs ) ) ;
if ( likely ( sbp ) ) {
nilfs_set_log_cursor ( sbp [ 0 ] , nilfs ) ;
nilfs_commit_super ( sbi , NILFS_SB_COMMIT ) ;
}
2010-06-28 12:49:32 +04:00
}
2009-07-22 20:26:33 +04:00
up_write ( & nilfs - > ns_sem ) ;
2009-04-07 06:01:35 +04:00
return err ;
}
2010-08-25 12:45:44 +04:00
int nilfs_attach_checkpoint ( struct nilfs_sb_info * sbi , __u64 cno , int curr_mnt ,
struct nilfs_root * * rootp )
2009-04-07 06:01:35 +04:00
{
struct the_nilfs * nilfs = sbi - > s_nilfs ;
2010-08-25 12:45:44 +04:00
struct nilfs_root * root ;
2009-04-07 06:01:35 +04:00
struct nilfs_checkpoint * raw_cp ;
struct buffer_head * bh_cp ;
2010-08-25 12:45:44 +04:00
int err = - ENOMEM ;
2009-04-07 06:01:35 +04:00
2010-08-25 12:45:44 +04:00
root = nilfs_find_or_create_root (
nilfs , curr_mnt ? NILFS_CPTREE_CURRENT_CNO : cno ) ;
if ( ! root )
return err ;
2009-04-07 06:01:35 +04:00
2010-08-14 08:07:15 +04:00
if ( root - > ifile )
goto reuse ; /* already attached checkpoint */
2009-04-07 06:01:35 +04:00
2009-08-18 10:58:24 +04:00
down_read ( & nilfs - > ns_segctor_sem ) ;
2009-04-07 06:01:35 +04:00
err = nilfs_cpfile_get_checkpoint ( nilfs - > ns_cpfile , cno , 0 , & raw_cp ,
& bh_cp ) ;
2009-08-18 10:58:24 +04:00
up_read ( & nilfs - > ns_segctor_sem ) ;
2009-04-07 06:01:35 +04:00
if ( unlikely ( err ) ) {
if ( err = = - ENOENT | | err = = - EINVAL ) {
printk ( KERN_ERR
" NILFS: Invalid checkpoint "
" (checkpoint number=%llu) \n " ,
( unsigned long long ) cno ) ;
err = - EINVAL ;
}
goto failed ;
}
2010-09-05 07:20:59 +04:00
err = nilfs_ifile_read ( sbi - > s_super , root , nilfs - > ns_inode_size ,
& raw_cp - > cp_ifile_inode , & root - > ifile ) ;
if ( err )
2009-04-07 06:01:35 +04:00
goto failed_bh ;
2010-08-14 09:48:32 +04:00
atomic_set ( & root - > inodes_count , le64_to_cpu ( raw_cp - > cp_inodes_count ) ) ;
atomic_set ( & root - > blocks_count , le64_to_cpu ( raw_cp - > cp_blocks_count ) ) ;
2009-04-07 06:01:35 +04:00
nilfs_cpfile_put_checkpoint ( nilfs - > ns_cpfile , cno , bh_cp ) ;
2010-08-25 12:45:44 +04:00
2010-08-14 08:07:15 +04:00
reuse :
2010-08-25 12:45:44 +04:00
* rootp = root ;
2009-04-07 06:01:35 +04:00
return 0 ;
failed_bh :
nilfs_cpfile_put_checkpoint ( nilfs - > ns_cpfile , cno , bh_cp ) ;
failed :
2010-08-25 12:45:44 +04:00
nilfs_put_root ( root ) ;
2009-04-07 06:01:35 +04:00
return err ;
}
2010-09-20 13:19:06 +04:00
static int nilfs_freeze ( struct super_block * sb )
{
struct nilfs_sb_info * sbi = NILFS_SB ( sb ) ;
struct the_nilfs * nilfs = sbi - > s_nilfs ;
int err ;
if ( sb - > s_flags & MS_RDONLY )
return 0 ;
2009-04-07 06:01:35 +04:00
2010-09-20 13:19:06 +04:00
/* Mark super block clean */
down_write ( & nilfs - > ns_sem ) ;
err = nilfs_cleanup_super ( sbi ) ;
up_write ( & nilfs - > ns_sem ) ;
2009-04-07 06:01:35 +04:00
return err ;
}
2010-09-20 13:19:06 +04:00
static int nilfs_unfreeze ( struct super_block * sb )
2009-04-07 06:01:35 +04:00
{
2010-09-20 13:19:06 +04:00
struct nilfs_sb_info * sbi = NILFS_SB ( sb ) ;
2009-04-07 06:01:35 +04:00
struct the_nilfs * nilfs = sbi - > s_nilfs ;
2010-09-20 13:19:06 +04:00
if ( sb - > s_flags & MS_RDONLY )
return 0 ;
down_write ( & nilfs - > ns_sem ) ;
nilfs_setup_super ( sbi , false ) ;
up_write ( & nilfs - > ns_sem ) ;
return 0 ;
2009-04-07 06:01:35 +04:00
}
static int nilfs_statfs ( struct dentry * dentry , struct kstatfs * buf )
{
struct super_block * sb = dentry - > d_sb ;
2010-08-14 09:48:32 +04:00
struct nilfs_root * root = NILFS_I ( dentry - > d_inode ) - > i_root ;
struct the_nilfs * nilfs = root - > nilfs ;
2009-03-26 04:16:57 +03:00
u64 id = huge_encode_dev ( sb - > s_bdev - > bd_dev ) ;
2009-04-07 06:01:35 +04:00
unsigned long long blocks ;
unsigned long overhead ;
unsigned long nrsvblocks ;
sector_t nfreeblocks ;
int err ;
/*
* Compute all of the segment blocks
*
* The blocks before first segment and after last segment
* are excluded .
*/
blocks = nilfs - > ns_blocks_per_segment * nilfs - > ns_nsegments
- nilfs - > ns_first_data_block ;
nrsvblocks = nilfs - > ns_nrsvsegs * nilfs - > ns_blocks_per_segment ;
/*
* Compute the overhead
*
2010-03-13 21:32:40 +03:00
* When distributing meta data blocks outside segment structure ,
2009-04-07 06:01:35 +04:00
* We must count them as the overhead .
*/
overhead = 0 ;
err = nilfs_count_free_blocks ( nilfs , & nfreeblocks ) ;
if ( unlikely ( err ) )
return err ;
buf - > f_type = NILFS_SUPER_MAGIC ;
buf - > f_bsize = sb - > s_blocksize ;
buf - > f_blocks = blocks - overhead ;
buf - > f_bfree = nfreeblocks ;
buf - > f_bavail = ( buf - > f_bfree > = nrsvblocks ) ?
( buf - > f_bfree - nrsvblocks ) : 0 ;
2010-08-14 09:48:32 +04:00
buf - > f_files = atomic_read ( & root - > inodes_count ) ;
2009-04-07 06:01:35 +04:00
buf - > f_ffree = 0 ; /* nilfs_count_free_inodes(sb); */
buf - > f_namelen = NILFS_NAME_LEN ;
2009-03-26 04:16:57 +03:00
buf - > f_fsid . val [ 0 ] = ( u32 ) id ;
buf - > f_fsid . val [ 1 ] = ( u32 ) ( id > > 32 ) ;
2009-04-07 06:01:35 +04:00
return 0 ;
}
2009-06-24 15:06:34 +04:00
static int nilfs_show_options ( struct seq_file * seq , struct vfsmount * vfs )
{
struct super_block * sb = vfs - > mnt_sb ;
struct nilfs_sb_info * sbi = NILFS_SB ( sb ) ;
2010-08-15 20:54:52 +04:00
struct nilfs_root * root = NILFS_I ( vfs - > mnt_root - > d_inode ) - > i_root ;
2009-06-24 15:06:34 +04:00
if ( ! nilfs_test_opt ( sbi , BARRIER ) )
2010-07-05 09:40:27 +04:00
seq_puts ( seq , " ,nobarrier " ) ;
2010-08-15 20:54:52 +04:00
if ( root - > cno ! = NILFS_CPTREE_CURRENT_CNO )
seq_printf ( seq , " ,cp=%llu " , ( unsigned long long ) root - > cno ) ;
2009-06-24 15:06:34 +04:00
if ( nilfs_test_opt ( sbi , ERRORS_PANIC ) )
2010-07-05 09:40:27 +04:00
seq_puts ( seq , " ,errors=panic " ) ;
2010-04-02 13:02:33 +04:00
if ( nilfs_test_opt ( sbi , ERRORS_CONT ) )
2010-07-05 09:40:27 +04:00
seq_puts ( seq , " ,errors=continue " ) ;
2009-06-24 15:06:34 +04:00
if ( nilfs_test_opt ( sbi , STRICT_ORDER ) )
2010-07-05 09:40:27 +04:00
seq_puts ( seq , " ,order=strict " ) ;
2009-11-19 21:28:01 +03:00
if ( nilfs_test_opt ( sbi , NORECOVERY ) )
2010-07-05 09:40:27 +04:00
seq_puts ( seq , " ,norecovery " ) ;
2010-01-30 12:06:35 +03:00
if ( nilfs_test_opt ( sbi , DISCARD ) )
2010-07-05 09:40:27 +04:00
seq_puts ( seq , " ,discard " ) ;
2009-06-24 15:06:34 +04:00
return 0 ;
}
2009-09-22 04:01:09 +04:00
static const struct super_operations nilfs_sops = {
2009-04-07 06:01:35 +04:00
. alloc_inode = nilfs_alloc_inode ,
. destroy_inode = nilfs_destroy_inode ,
. dirty_inode = nilfs_dirty_inode ,
/* .write_inode = nilfs_write_inode, */
/* .put_inode = nilfs_put_inode, */
/* .drop_inode = nilfs_drop_inode, */
2010-06-07 19:55:00 +04:00
. evict_inode = nilfs_evict_inode ,
2009-04-07 06:01:35 +04:00
. put_super = nilfs_put_super ,
2009-07-22 20:33:49 +04:00
/* .write_super = nilfs_write_super, */
2009-04-07 06:01:35 +04:00
. sync_fs = nilfs_sync_fs ,
2010-09-20 13:19:06 +04:00
. freeze_fs = nilfs_freeze ,
. unfreeze_fs = nilfs_unfreeze ,
2009-04-07 06:01:35 +04:00
/* .write_super_lockfs */
/* .unlockfs */
. statfs = nilfs_statfs ,
. remount_fs = nilfs_remount ,
/* .umount_begin */
2009-06-24 15:06:34 +04:00
. show_options = nilfs_show_options
2009-04-07 06:01:35 +04:00
} ;
enum {
Opt_err_cont , Opt_err_panic , Opt_err_ro ,
2010-07-05 08:00:08 +04:00
Opt_barrier , Opt_nobarrier , Opt_snapshot , Opt_order , Opt_norecovery ,
2010-07-05 09:27:04 +04:00
Opt_discard , Opt_nodiscard , Opt_err ,
2009-04-07 06:01:35 +04:00
} ;
static match_table_t tokens = {
{ Opt_err_cont , " errors=continue " } ,
{ Opt_err_panic , " errors=panic " } ,
{ Opt_err_ro , " errors=remount-ro " } ,
2010-07-05 08:00:08 +04:00
{ Opt_barrier , " barrier " } ,
2009-11-12 08:07:26 +03:00
{ Opt_nobarrier , " nobarrier " } ,
2009-04-07 06:01:35 +04:00
{ Opt_snapshot , " cp=%u " } ,
{ Opt_order , " order=%s " } ,
2009-11-19 21:28:01 +03:00
{ Opt_norecovery , " norecovery " } ,
2010-01-30 12:06:35 +03:00
{ Opt_discard , " discard " } ,
2010-07-05 09:27:04 +04:00
{ Opt_nodiscard , " nodiscard " } ,
2009-04-07 06:01:35 +04:00
{ Opt_err , NULL }
} ;
2010-07-05 15:08:33 +04:00
static int parse_options ( char * options , struct super_block * sb , int is_remount )
2009-04-07 06:01:35 +04:00
{
struct nilfs_sb_info * sbi = NILFS_SB ( sb ) ;
char * p ;
substring_t args [ MAX_OPT_ARGS ] ;
if ( ! options )
return 1 ;
while ( ( p = strsep ( & options , " , " ) ) ! = NULL ) {
int token ;
if ( ! * p )
continue ;
token = match_token ( p , tokens , args ) ;
switch ( token ) {
2010-07-05 08:00:08 +04:00
case Opt_barrier :
nilfs_set_opt ( sbi , BARRIER ) ;
break ;
2009-11-12 08:07:26 +03:00
case Opt_nobarrier :
nilfs_clear_opt ( sbi , BARRIER ) ;
2009-04-07 06:01:35 +04:00
break ;
case Opt_order :
if ( strcmp ( args [ 0 ] . from , " relaxed " ) = = 0 )
/* Ordered data semantics */
nilfs_clear_opt ( sbi , STRICT_ORDER ) ;
else if ( strcmp ( args [ 0 ] . from , " strict " ) = = 0 )
/* Strict in-order semantics */
nilfs_set_opt ( sbi , STRICT_ORDER ) ;
else
return 0 ;
break ;
case Opt_err_panic :
nilfs_write_opt ( sbi , ERROR_MODE , ERRORS_PANIC ) ;
break ;
case Opt_err_ro :
nilfs_write_opt ( sbi , ERROR_MODE , ERRORS_RO ) ;
break ;
case Opt_err_cont :
nilfs_write_opt ( sbi , ERROR_MODE , ERRORS_CONT ) ;
break ;
case Opt_snapshot :
2010-07-05 15:08:33 +04:00
if ( is_remount ) {
2010-08-15 20:54:52 +04:00
printk ( KERN_ERR
" NILFS: \" %s \" option is invalid "
" for remount. \n " , p ) ;
2009-04-07 06:01:35 +04:00
return 0 ;
2010-07-05 15:08:33 +04:00
}
2009-04-07 06:01:35 +04:00
break ;
2009-11-19 21:28:01 +03:00
case Opt_norecovery :
nilfs_set_opt ( sbi , NORECOVERY ) ;
break ;
2010-01-30 12:06:35 +03:00
case Opt_discard :
nilfs_set_opt ( sbi , DISCARD ) ;
break ;
2010-07-05 09:27:04 +04:00
case Opt_nodiscard :
nilfs_clear_opt ( sbi , DISCARD ) ;
break ;
2009-04-07 06:01:35 +04:00
default :
printk ( KERN_ERR
" NILFS: Unrecognized mount option \" %s \" \n " , p ) ;
return 0 ;
}
}
return 1 ;
}
static inline void
nilfs_set_default_options ( struct nilfs_sb_info * sbi ,
struct nilfs_super_block * sbp )
{
sbi - > s_mount_opt =
2010-04-02 13:02:33 +04:00
NILFS_MOUNT_ERRORS_RO | NILFS_MOUNT_BARRIER ;
2009-04-07 06:01:35 +04:00
}
2010-09-20 13:19:06 +04:00
static int nilfs_setup_super ( struct nilfs_sb_info * sbi , int is_mount )
2009-04-07 06:01:35 +04:00
{
struct the_nilfs * nilfs = sbi - > s_nilfs ;
2010-06-28 12:49:32 +04:00
struct nilfs_super_block * * sbp ;
int max_mnt_count ;
int mnt_count ;
/* nilfs->ns_sem must be locked by the caller. */
2010-06-28 12:49:33 +04:00
sbp = nilfs_prepare_super ( sbi , 0 ) ;
2010-06-28 12:49:32 +04:00
if ( ! sbp )
return - EIO ;
2010-09-20 13:19:06 +04:00
if ( ! is_mount )
goto skip_mount_setup ;
2010-06-28 12:49:32 +04:00
max_mnt_count = le16_to_cpu ( sbp [ 0 ] - > s_max_mnt_count ) ;
mnt_count = le16_to_cpu ( sbp [ 0 ] - > s_mnt_count ) ;
2009-04-07 06:01:35 +04:00
2009-11-19 10:58:40 +03:00
if ( nilfs - > ns_mount_state & NILFS_ERROR_FS ) {
2009-04-07 06:01:35 +04:00
printk ( KERN_WARNING
" NILFS warning: mounting fs with errors \n " ) ;
#if 0
} else if ( max_mnt_count > = 0 & & mnt_count > = max_mnt_count ) {
printk ( KERN_WARNING
" NILFS warning: maximal mount count reached \n " ) ;
# endif
}
if ( ! max_mnt_count )
2010-06-28 12:49:32 +04:00
sbp [ 0 ] - > s_max_mnt_count = cpu_to_le16 ( NILFS_DFL_MAX_MNT_COUNT ) ;
2009-04-07 06:01:35 +04:00
2010-06-28 12:49:32 +04:00
sbp [ 0 ] - > s_mnt_count = cpu_to_le16 ( mnt_count + 1 ) ;
2010-09-20 13:19:06 +04:00
sbp [ 0 ] - > s_mtime = cpu_to_le64 ( get_seconds ( ) ) ;
skip_mount_setup :
2010-06-28 12:49:32 +04:00
sbp [ 0 ] - > s_state =
cpu_to_le16 ( le16_to_cpu ( sbp [ 0 ] - > s_state ) & ~ NILFS_VALID_FS ) ;
2010-06-28 12:49:33 +04:00
/* synchronize sbp[1] with sbp[0] */
memcpy ( sbp [ 1 ] , sbp [ 0 ] , nilfs - > ns_sbsize ) ;
return nilfs_commit_super ( sbi , NILFS_SB_COMMIT_ALL ) ;
2009-04-07 06:01:35 +04:00
}
2009-04-07 06:01:59 +04:00
struct nilfs_super_block * nilfs_read_super_block ( struct super_block * sb ,
u64 pos , int blocksize ,
struct buffer_head * * pbh )
2009-04-07 06:01:35 +04:00
{
2009-04-07 06:01:59 +04:00
unsigned long long sb_index = pos ;
unsigned long offset ;
2009-04-07 06:01:35 +04:00
2009-04-07 06:01:59 +04:00
offset = do_div ( sb_index , blocksize ) ;
2009-04-07 06:01:35 +04:00
* pbh = sb_bread ( sb , sb_index ) ;
2009-04-07 06:01:59 +04:00
if ( ! * pbh )
2009-04-07 06:01:35 +04:00
return NULL ;
return ( struct nilfs_super_block * ) ( ( char * ) ( * pbh ) - > b_data + offset ) ;
}
int nilfs_store_magic_and_option ( struct super_block * sb ,
struct nilfs_super_block * sbp ,
char * data )
{
struct nilfs_sb_info * sbi = NILFS_SB ( sb ) ;
sb - > s_magic = le16_to_cpu ( sbp - > s_magic ) ;
/* FS independent flags */
# ifdef NILFS_ATIME_DISABLE
sb - > s_flags | = MS_NOATIME ;
# endif
nilfs_set_default_options ( sbi , sbp ) ;
sbi - > s_resuid = le16_to_cpu ( sbp - > s_def_resuid ) ;
sbi - > s_resgid = le16_to_cpu ( sbp - > s_def_resgid ) ;
sbi - > s_interval = le32_to_cpu ( sbp - > s_c_interval ) ;
sbi - > s_watermark = le32_to_cpu ( sbp - > s_c_block_max ) ;
2010-07-05 15:08:33 +04:00
return ! parse_options ( data , sb , 0 ) ? - EINVAL : 0 ;
2009-04-07 06:01:35 +04:00
}
2010-07-21 22:22:20 +04:00
int nilfs_check_feature_compatibility ( struct super_block * sb ,
struct nilfs_super_block * sbp )
{
__u64 features ;
features = le64_to_cpu ( sbp - > s_feature_incompat ) &
~ NILFS_FEATURE_INCOMPAT_SUPP ;
if ( features ) {
printk ( KERN_ERR " NILFS: couldn't mount because of unsupported "
" optional features (%llx) \n " ,
( unsigned long long ) features ) ;
return - EINVAL ;
}
features = le64_to_cpu ( sbp - > s_feature_compat_ro ) &
~ NILFS_FEATURE_COMPAT_RO_SUPP ;
if ( ! ( sb - > s_flags & MS_RDONLY ) & & features ) {
printk ( KERN_ERR " NILFS: couldn't mount RDWR because of "
" unsupported optional features (%llx) \n " ,
( unsigned long long ) features ) ;
return - EINVAL ;
}
return 0 ;
}
2010-08-25 20:52:51 +04:00
static int nilfs_get_root_dentry ( struct super_block * sb ,
struct nilfs_root * root ,
struct dentry * * root_dentry )
{
struct inode * inode ;
struct dentry * dentry ;
int ret = 0 ;
inode = nilfs_iget ( sb , root , NILFS_ROOT_INO ) ;
if ( IS_ERR ( inode ) ) {
printk ( KERN_ERR " NILFS: get root inode failed \n " ) ;
ret = PTR_ERR ( inode ) ;
goto out ;
}
if ( ! S_ISDIR ( inode - > i_mode ) | | ! inode - > i_blocks | | ! inode - > i_size ) {
iput ( inode ) ;
printk ( KERN_ERR " NILFS: corrupt root inode. \n " ) ;
ret = - EINVAL ;
goto out ;
}
2010-08-15 20:54:52 +04:00
if ( root - > cno = = NILFS_CPTREE_CURRENT_CNO ) {
dentry = d_find_alias ( inode ) ;
if ( ! dentry ) {
dentry = d_alloc_root ( inode ) ;
if ( ! dentry ) {
iput ( inode ) ;
ret = - ENOMEM ;
goto failed_dentry ;
}
} else {
iput ( inode ) ;
}
} else {
dentry = d_obtain_alias ( inode ) ;
if ( IS_ERR ( dentry ) ) {
ret = PTR_ERR ( dentry ) ;
goto failed_dentry ;
}
2010-08-25 20:52:51 +04:00
}
* root_dentry = dentry ;
out :
return ret ;
2010-08-15 20:54:52 +04:00
failed_dentry :
printk ( KERN_ERR " NILFS: get root dentry failed \n " ) ;
goto out ;
2010-08-25 20:52:51 +04:00
}
2010-08-25 21:15:41 +04:00
static int nilfs_attach_snapshot ( struct super_block * s , __u64 cno ,
struct dentry * * root_dentry )
{
struct the_nilfs * nilfs = NILFS_SB ( s ) - > s_nilfs ;
struct nilfs_root * root ;
int ret ;
down_read ( & nilfs - > ns_segctor_sem ) ;
ret = nilfs_cpfile_is_snapshot ( nilfs - > ns_cpfile , cno ) ;
up_read ( & nilfs - > ns_segctor_sem ) ;
if ( ret < 0 ) {
ret = ( ret = = - ENOENT ) ? - EINVAL : ret ;
goto out ;
} else if ( ! ret ) {
printk ( KERN_ERR " NILFS: The specified checkpoint is "
" not a snapshot (checkpoint number=%llu). \n " ,
( unsigned long long ) cno ) ;
ret = - EINVAL ;
goto out ;
}
ret = nilfs_attach_checkpoint ( NILFS_SB ( s ) , cno , false , & root ) ;
if ( ret ) {
printk ( KERN_ERR " NILFS: error loading snapshot "
" (checkpoint number=%llu). \n " ,
( unsigned long long ) cno ) ;
goto out ;
}
ret = nilfs_get_root_dentry ( s , root , root_dentry ) ;
nilfs_put_root ( root ) ;
out :
return ret ;
}
2010-08-15 20:54:52 +04:00
static int nilfs_tree_was_touched ( struct dentry * root_dentry )
{
return atomic_read ( & root_dentry - > d_count ) > 1 ;
}
/**
* nilfs_try_to_shrink_tree ( ) - try to shrink dentries of a checkpoint
* @ root_dentry : root dentry of the tree to be shrunk
*
* This function returns true if the tree was in - use .
*/
static int nilfs_try_to_shrink_tree ( struct dentry * root_dentry )
{
if ( have_submounts ( root_dentry ) )
return true ;
shrink_dcache_parent ( root_dentry ) ;
return nilfs_tree_was_touched ( root_dentry ) ;
}
2010-09-13 06:16:34 +04:00
int nilfs_checkpoint_is_mounted ( struct super_block * sb , __u64 cno )
{
struct the_nilfs * nilfs = NILFS_SB ( sb ) - > s_nilfs ;
struct nilfs_root * root ;
struct inode * inode ;
struct dentry * dentry ;
int ret ;
if ( cno < 0 | | cno > nilfs - > ns_cno )
return false ;
if ( cno > = nilfs_last_cno ( nilfs ) )
return true ; /* protect recent checkpoints */
ret = false ;
root = nilfs_lookup_root ( NILFS_SB ( sb ) - > s_nilfs , cno ) ;
if ( root ) {
inode = nilfs_ilookup ( sb , root , NILFS_ROOT_INO ) ;
if ( inode ) {
dentry = d_find_alias ( inode ) ;
if ( dentry ) {
if ( nilfs_tree_was_touched ( dentry ) )
ret = nilfs_try_to_shrink_tree ( dentry ) ;
dput ( dentry ) ;
}
iput ( inode ) ;
}
nilfs_put_root ( root ) ;
}
return ret ;
}
2009-04-07 06:01:35 +04:00
/**
* nilfs_fill_super ( ) - initialize a super block instance
* @ sb : super_block
* @ data : mount options
* @ silent : silent mode flag
*
2009-06-07 20:39:33 +04:00
* This function is called exclusively by nilfs - > ns_mount_mutex .
2009-04-07 06:01:35 +04:00
* So , the recovery process is protected from other simultaneous mounts .
*/
static int
2010-09-08 21:07:56 +04:00
nilfs_fill_super ( struct super_block * sb , void * data , int silent )
2009-04-07 06:01:35 +04:00
{
2010-09-08 21:07:56 +04:00
struct the_nilfs * nilfs ;
2009-04-07 06:01:35 +04:00
struct nilfs_sb_info * sbi ;
2010-08-25 12:45:44 +04:00
struct nilfs_root * fsroot ;
2010-10-07 09:19:48 +04:00
struct backing_dev_info * bdi ;
2009-04-07 06:01:35 +04:00
__u64 cno ;
int err ;
sbi = kzalloc ( sizeof ( * sbi ) , GFP_KERNEL ) ;
if ( ! sbi )
return - ENOMEM ;
sb - > s_fs_info = sbi ;
2010-09-08 21:07:56 +04:00
sbi - > s_super = sb ;
2009-04-07 06:01:35 +04:00
2010-09-08 21:07:56 +04:00
nilfs = alloc_nilfs ( sb - > s_bdev ) ;
if ( ! nilfs ) {
err = - ENOMEM ;
goto failed_sbi ;
}
2009-04-07 06:01:35 +04:00
sbi - > s_nilfs = nilfs ;
err = init_nilfs ( nilfs , sbi , ( char * ) data ) ;
if ( err )
2010-09-08 21:07:56 +04:00
goto failed_nilfs ;
2009-04-07 06:01:35 +04:00
spin_lock_init ( & sbi - > s_inode_lock ) ;
INIT_LIST_HEAD ( & sbi - > s_dirty_files ) ;
/*
* Following initialization is overlapped because
* nilfs_sb_info structure has been cleared at the beginning .
* But we reserve them to keep our interest and make ready
* for the future change .
*/
get_random_bytes ( & sbi - > s_next_generation ,
sizeof ( sbi - > s_next_generation ) ) ;
spin_lock_init ( & sbi - > s_next_gen_lock ) ;
sb - > s_op = & nilfs_sops ;
sb - > s_export_op = & nilfs_export_ops ;
sb - > s_root = NULL ;
2009-04-07 06:02:00 +04:00
sb - > s_time_gran = 1 ;
2010-10-07 09:19:48 +04:00
bdi = sb - > s_bdev - > bd_inode - > i_mapping - > backing_dev_info ;
sb - > s_bdi = bdi ? : & default_backing_dev_info ;
2009-04-07 06:01:35 +04:00
2009-11-19 10:58:40 +03:00
err = load_nilfs ( nilfs , sbi ) ;
if ( err )
2010-09-08 21:07:56 +04:00
goto failed_nilfs ;
2009-11-19 10:58:40 +03:00
2009-04-07 06:01:35 +04:00
cno = nilfs_last_cno ( nilfs ) ;
2010-08-25 21:15:41 +04:00
err = nilfs_attach_checkpoint ( sbi , cno , true , & fsroot ) ;
2009-04-07 06:01:35 +04:00
if ( err ) {
2010-08-15 20:54:52 +04:00
printk ( KERN_ERR " NILFS: error loading last checkpoint "
" (checkpoint number=%llu). \n " , ( unsigned long long ) cno ) ;
2010-09-05 07:20:59 +04:00
goto failed_unload ;
2009-04-07 06:01:35 +04:00
}
if ( ! ( sb - > s_flags & MS_RDONLY ) ) {
2010-08-14 08:07:15 +04:00
err = nilfs_attach_segment_constructor ( sbi , fsroot ) ;
2009-04-07 06:01:35 +04:00
if ( err )
goto failed_checkpoint ;
}
2010-08-25 20:52:51 +04:00
err = nilfs_get_root_dentry ( sb , fsroot , & sb - > s_root ) ;
if ( err )
2009-04-07 06:01:35 +04:00
goto failed_segctor ;
2010-08-25 12:45:44 +04:00
nilfs_put_root ( fsroot ) ;
2009-04-07 06:01:35 +04:00
if ( ! ( sb - > s_flags & MS_RDONLY ) ) {
down_write ( & nilfs - > ns_sem ) ;
2010-09-20 13:19:06 +04:00
nilfs_setup_super ( sbi , true ) ;
2009-04-07 06:01:35 +04:00
up_write ( & nilfs - > ns_sem ) ;
}
return 0 ;
failed_segctor :
nilfs_detach_segment_constructor ( sbi ) ;
failed_checkpoint :
2010-08-25 12:45:44 +04:00
nilfs_put_root ( fsroot ) ;
2009-04-07 06:01:35 +04:00
2010-09-05 07:20:59 +04:00
failed_unload :
iput ( nilfs - > ns_sufile ) ;
iput ( nilfs - > ns_cpfile ) ;
iput ( nilfs - > ns_dat ) ;
2010-09-08 21:07:56 +04:00
failed_nilfs :
destroy_nilfs ( nilfs ) ;
2009-04-07 06:01:35 +04:00
failed_sbi :
sb - > s_fs_info = NULL ;
2010-08-15 20:54:52 +04:00
kfree ( sbi ) ;
2009-04-07 06:01:35 +04:00
return err ;
}
static int nilfs_remount ( struct super_block * sb , int * flags , char * data )
{
struct nilfs_sb_info * sbi = NILFS_SB ( sb ) ;
struct the_nilfs * nilfs = sbi - > s_nilfs ;
unsigned long old_sb_flags ;
struct nilfs_mount_options old_opts ;
2010-08-15 20:54:52 +04:00
int err ;
2009-04-07 06:01:35 +04:00
old_sb_flags = sb - > s_flags ;
old_opts . mount_opt = sbi - > s_mount_opt ;
2010-07-05 15:08:33 +04:00
if ( ! parse_options ( data , sb , 1 ) ) {
2009-04-07 06:01:35 +04:00
err = - EINVAL ;
goto restore_opts ;
}
sb - > s_flags = ( sb - > s_flags & ~ MS_POSIXACL ) ;
2010-05-09 16:51:53 +04:00
err = - EINVAL ;
2009-04-07 06:01:35 +04:00
2009-11-19 21:28:01 +03:00
if ( ! nilfs_valid_fs ( nilfs ) ) {
printk ( KERN_WARNING " NILFS (device %s): couldn't "
" remount because the filesystem is in an "
" incomplete recovery state. \n " , sb - > s_id ) ;
goto restore_opts ;
}
2009-04-07 06:01:35 +04:00
if ( ( * flags & MS_RDONLY ) = = ( sb - > s_flags & MS_RDONLY ) )
goto out ;
if ( * flags & MS_RDONLY ) {
/* Shutting down the segment constructor */
nilfs_detach_segment_constructor ( sbi ) ;
sb - > s_flags | = MS_RDONLY ;
/*
* Remounting a valid RW partition RDONLY , so set
* the RDONLY flag and then mark the partition as valid again .
*/
down_write ( & nilfs - > ns_sem ) ;
2010-06-28 12:49:29 +04:00
nilfs_cleanup_super ( sbi ) ;
2009-04-07 06:01:35 +04:00
up_write ( & nilfs - > ns_sem ) ;
} else {
2010-07-21 22:22:20 +04:00
__u64 features ;
2010-08-14 08:07:15 +04:00
struct nilfs_root * root ;
2010-07-21 22:22:20 +04:00
2009-04-07 06:01:35 +04:00
/*
* Mounting a RDONLY partition read - write , so reread and
* store the current valid flag . ( It may have been changed
* by fsck since we originally mounted the partition . )
*/
2010-07-21 22:22:20 +04:00
down_read ( & nilfs - > ns_sem ) ;
features = le64_to_cpu ( nilfs - > ns_sbp [ 0 ] - > s_feature_compat_ro ) &
~ NILFS_FEATURE_COMPAT_RO_SUPP ;
up_read ( & nilfs - > ns_sem ) ;
if ( features ) {
printk ( KERN_WARNING " NILFS (device %s): couldn't "
" remount RDWR because of unsupported optional "
" features (%llx) \n " ,
sb - > s_id , ( unsigned long long ) features ) ;
err = - EROFS ;
goto restore_opts ;
}
2009-04-07 06:01:35 +04:00
sb - > s_flags & = ~ MS_RDONLY ;
2010-08-14 08:07:15 +04:00
root = NILFS_I ( sb - > s_root - > d_inode ) - > i_root ;
err = nilfs_attach_segment_constructor ( sbi , root ) ;
2009-04-07 06:01:35 +04:00
if ( err )
2009-06-07 20:39:32 +04:00
goto restore_opts ;
2009-04-07 06:01:35 +04:00
down_write ( & nilfs - > ns_sem ) ;
2010-09-20 13:19:06 +04:00
nilfs_setup_super ( sbi , true ) ;
2009-04-07 06:01:35 +04:00
up_write ( & nilfs - > ns_sem ) ;
}
out :
return 0 ;
restore_opts :
sb - > s_flags = old_sb_flags ;
sbi - > s_mount_opt = old_opts . mount_opt ;
return err ;
}
struct nilfs_super_data {
struct block_device * bdev ;
2009-06-07 20:39:31 +04:00
struct nilfs_sb_info * sbi ;
2009-04-07 06:01:35 +04:00
__u64 cno ;
int flags ;
} ;
/**
* nilfs_identify - pre - read mount options needed to identify mount instance
* @ data : mount options
* @ sd : nilfs_super_data
*/
static int nilfs_identify ( char * data , struct nilfs_super_data * sd )
{
char * p , * options = data ;
substring_t args [ MAX_OPT_ARGS ] ;
2010-09-15 19:36:24 +04:00
int token ;
2009-04-07 06:01:35 +04:00
int ret = 0 ;
do {
p = strsep ( & options , " , " ) ;
if ( p ! = NULL & & * p ) {
token = match_token ( p , tokens , args ) ;
if ( token = = Opt_snapshot ) {
2010-09-15 19:36:24 +04:00
if ( ! ( sd - > flags & MS_RDONLY ) ) {
2009-04-07 06:01:35 +04:00
ret + + ;
2010-09-15 19:36:24 +04:00
} else {
sd - > cno = simple_strtoull ( args [ 0 ] . from ,
NULL , 0 ) ;
/*
* No need to see the end pointer ;
* match_token ( ) has done syntax
* checking .
*/
if ( sd - > cno = = 0 )
ret + + ;
2009-04-07 06:01:35 +04:00
}
}
if ( ret )
printk ( KERN_ERR
" NILFS: invalid mount option: %s \n " , p ) ;
}
if ( ! options )
break ;
BUG_ON ( options = = data ) ;
* ( options - 1 ) = ' , ' ;
} while ( ! ret ) ;
return ret ;
}
static int nilfs_set_bdev_super ( struct super_block * s , void * data )
{
2010-08-15 20:54:52 +04:00
s - > s_bdev = data ;
2009-04-07 06:01:35 +04:00
s - > s_dev = s - > s_bdev - > bd_dev ;
return 0 ;
}
static int nilfs_test_bdev_super ( struct super_block * s , void * data )
{
2010-08-15 20:54:52 +04:00
return ( void * ) s - > s_bdev = = data ;
2009-04-07 06:01:35 +04:00
}
2010-07-26 12:19:34 +04:00
static struct dentry *
nilfs_mount ( struct file_system_type * fs_type , int flags ,
const char * dev_name , void * data )
2009-04-07 06:01:35 +04:00
{
struct nilfs_super_data sd ;
2009-06-07 20:39:29 +04:00
struct super_block * s ;
2010-05-08 21:57:57 +04:00
fmode_t mode = FMODE_READ ;
2010-08-15 20:54:52 +04:00
struct dentry * root_dentry ;
int err , s_new = false ;
2009-04-07 06:01:35 +04:00
2010-05-08 21:57:57 +04:00
if ( ! ( flags & MS_RDONLY ) )
mode | = FMODE_WRITE ;
sd . bdev = open_bdev_exclusive ( dev_name , mode , fs_type ) ;
2010-02-24 15:25:32 +03:00
if ( IS_ERR ( sd . bdev ) )
2010-07-26 12:19:34 +04:00
return ERR_CAST ( sd . bdev ) ;
2009-04-07 06:01:35 +04:00
sd . cno = 0 ;
sd . flags = flags ;
if ( nilfs_identify ( ( char * ) data , & sd ) ) {
err = - EINVAL ;
goto failed ;
}
2009-06-07 20:39:31 +04:00
/*
2010-09-20 13:19:06 +04:00
* once the super is inserted into the list by sget , s_umount
* will protect the lockfs code from trying to start a snapshot
* while we are mounting
2009-06-07 20:39:31 +04:00
*/
2010-09-20 13:19:06 +04:00
mutex_lock ( & sd . bdev - > bd_fsfreeze_mutex ) ;
if ( sd . bdev - > bd_fsfreeze_count > 0 ) {
mutex_unlock ( & sd . bdev - > bd_fsfreeze_mutex ) ;
err = - EBUSY ;
goto failed ;
}
2010-08-15 20:54:52 +04:00
s = sget ( fs_type , nilfs_test_bdev_super , nilfs_set_bdev_super , sd . bdev ) ;
2010-09-20 13:19:06 +04:00
mutex_unlock ( & sd . bdev - > bd_fsfreeze_mutex ) ;
2009-06-07 20:39:29 +04:00
if ( IS_ERR ( s ) ) {
err = PTR_ERR ( s ) ;
2010-09-08 21:07:56 +04:00
goto failed ;
2009-04-07 06:01:35 +04:00
}
if ( ! s - > s_root ) {
char b [ BDEVNAME_SIZE ] ;
2010-08-15 20:54:52 +04:00
s_new = true ;
2009-06-07 20:39:29 +04:00
/* New superblock instance created */
2009-04-07 06:01:35 +04:00
s - > s_flags = flags ;
2010-05-08 22:01:32 +04:00
s - > s_mode = mode ;
2009-04-07 06:01:35 +04:00
strlcpy ( s - > s_id , bdevname ( sd . bdev , b ) , sizeof ( s - > s_id ) ) ;
sb_set_blocksize ( s , block_size ( sd . bdev ) ) ;
2010-09-08 21:07:56 +04:00
err = nilfs_fill_super ( s , data , flags & MS_SILENT ? 1 : 0 ) ;
2009-04-07 06:01:35 +04:00
if ( err )
2010-09-08 21:07:56 +04:00
goto failed_super ;
2009-04-07 06:01:35 +04:00
s - > s_flags | = MS_ACTIVE ;
2010-08-15 20:54:52 +04:00
} else if ( ! sd . cno ) {
int busy = false ;
if ( nilfs_tree_was_touched ( s - > s_root ) ) {
busy = nilfs_try_to_shrink_tree ( s - > s_root ) ;
if ( busy & & ( flags ^ s - > s_flags ) & MS_RDONLY ) {
printk ( KERN_ERR " NILFS: the device already "
" has a %s mount. \n " ,
( s - > s_flags & MS_RDONLY ) ?
" read-only " : " read/write " ) ;
err = - EBUSY ;
goto failed_super ;
}
}
if ( ! busy ) {
/*
* Try remount to setup mount states if the current
* tree is not mounted and only snapshots use this sb .
*/
err = nilfs_remount ( s , & flags , data ) ;
if ( err )
goto failed_super ;
}
2009-04-07 06:01:35 +04:00
}
2010-08-15 20:54:52 +04:00
if ( sd . cno ) {
err = nilfs_attach_snapshot ( s , sd . cno , & root_dentry ) ;
2010-09-08 21:07:56 +04:00
if ( err )
2010-08-15 20:54:52 +04:00
goto failed_super ;
} else {
root_dentry = dget ( s - > s_root ) ;
2009-04-07 06:01:35 +04:00
}
2010-08-15 20:54:52 +04:00
if ( ! s_new )
2010-05-08 21:57:57 +04:00
close_bdev_exclusive ( sd . bdev , mode ) ;
2009-04-07 06:01:35 +04:00
2010-07-26 12:19:34 +04:00
return root_dentry ;
2009-04-07 06:01:35 +04:00
2010-08-15 20:54:52 +04:00
failed_super :
2009-08-09 00:52:02 +04:00
deactivate_locked_super ( s ) ;
2009-04-07 06:01:35 +04:00
2010-09-08 21:07:56 +04:00
failed :
if ( ! s_new )
close_bdev_exclusive ( sd . bdev , mode ) ;
2010-07-26 12:19:34 +04:00
return ERR_PTR ( err ) ;
2009-04-07 06:01:35 +04:00
}
struct file_system_type nilfs_fs_type = {
. owner = THIS_MODULE ,
. name = " nilfs2 " ,
2010-07-26 12:19:34 +04:00
. mount = nilfs_mount ,
2009-04-07 06:01:35 +04:00
. kill_sb = kill_block_super ,
. fs_flags = FS_REQUIRES_DEV ,
} ;
2010-04-05 20:54:11 +04:00
static void nilfs_inode_init_once ( void * obj )
2009-04-07 06:01:35 +04:00
{
2010-04-05 20:54:11 +04:00
struct nilfs_inode_info * ii = obj ;
2009-04-07 06:01:35 +04:00
2010-04-05 20:54:11 +04:00
INIT_LIST_HEAD ( & ii - > i_dirty ) ;
# ifdef CONFIG_NILFS_XATTR
init_rwsem ( & ii - > xattr_sem ) ;
# endif
nilfs_btnode_cache_init_once ( & ii - > i_btnode_cache ) ;
2010-07-10 15:52:09 +04:00
ii - > i_bmap = & ii - > i_bmap_data ;
2010-04-05 20:54:11 +04:00
inode_init_once ( & ii - > vfs_inode ) ;
}
2009-04-07 06:01:35 +04:00
2010-04-05 20:54:11 +04:00
static void nilfs_segbuf_init_once ( void * obj )
{
memset ( obj , 0 , sizeof ( struct nilfs_segment_buffer ) ) ;
}
2009-04-07 06:01:35 +04:00
2010-04-05 20:54:11 +04:00
static void nilfs_destroy_cachep ( void )
{
2010-05-22 07:49:32 +04:00
if ( nilfs_inode_cachep )
2010-04-05 20:54:11 +04:00
kmem_cache_destroy ( nilfs_inode_cachep ) ;
2010-05-22 07:49:32 +04:00
if ( nilfs_transaction_cachep )
2010-04-05 20:54:11 +04:00
kmem_cache_destroy ( nilfs_transaction_cachep ) ;
2010-05-22 07:49:32 +04:00
if ( nilfs_segbuf_cachep )
2010-04-05 20:54:11 +04:00
kmem_cache_destroy ( nilfs_segbuf_cachep ) ;
2010-05-22 07:49:32 +04:00
if ( nilfs_btree_path_cache )
2010-04-05 20:54:11 +04:00
kmem_cache_destroy ( nilfs_btree_path_cache ) ;
}
2009-04-07 06:01:35 +04:00
2010-04-05 20:54:11 +04:00
static int __init nilfs_init_cachep ( void )
{
nilfs_inode_cachep = kmem_cache_create ( " nilfs2_inode_cache " ,
sizeof ( struct nilfs_inode_info ) , 0 ,
SLAB_RECLAIM_ACCOUNT , nilfs_inode_init_once ) ;
if ( ! nilfs_inode_cachep )
goto fail ;
nilfs_transaction_cachep = kmem_cache_create ( " nilfs2_transaction_cache " ,
sizeof ( struct nilfs_transaction_info ) , 0 ,
SLAB_RECLAIM_ACCOUNT , NULL ) ;
if ( ! nilfs_transaction_cachep )
goto fail ;
nilfs_segbuf_cachep = kmem_cache_create ( " nilfs2_segbuf_cache " ,
sizeof ( struct nilfs_segment_buffer ) , 0 ,
SLAB_RECLAIM_ACCOUNT , nilfs_segbuf_init_once ) ;
if ( ! nilfs_segbuf_cachep )
goto fail ;
nilfs_btree_path_cache = kmem_cache_create ( " nilfs2_btree_path_cache " ,
sizeof ( struct nilfs_btree_path ) * NILFS_BTREE_LEVEL_MAX ,
0 , 0 , NULL ) ;
if ( ! nilfs_btree_path_cache )
goto fail ;
2009-04-07 06:01:35 +04:00
return 0 ;
2010-04-05 20:54:11 +04:00
fail :
nilfs_destroy_cachep ( ) ;
return - ENOMEM ;
}
static int __init init_nilfs_fs ( void )
{
int err ;
2009-04-07 06:01:35 +04:00
2010-04-05 20:54:11 +04:00
err = nilfs_init_cachep ( ) ;
if ( err )
goto fail ;
2009-04-07 06:01:35 +04:00
2010-04-05 20:54:11 +04:00
err = register_filesystem ( & nilfs_fs_type ) ;
if ( err )
goto free_cachep ;
2009-04-07 06:01:35 +04:00
2010-04-09 19:09:53 +04:00
printk ( KERN_INFO " NILFS version 2 loaded \n " ) ;
2010-04-05 20:54:11 +04:00
return 0 ;
2009-04-07 06:01:35 +04:00
2010-04-05 20:54:11 +04:00
free_cachep :
nilfs_destroy_cachep ( ) ;
fail :
2009-04-07 06:01:35 +04:00
return err ;
}
static void __exit exit_nilfs_fs ( void )
{
2010-04-05 20:54:11 +04:00
nilfs_destroy_cachep ( ) ;
2009-04-07 06:01:35 +04:00
unregister_filesystem ( & nilfs_fs_type ) ;
}
module_init ( init_nilfs_fs )
module_exit ( exit_nilfs_fs )