2009-04-07 06:01:33 +04:00
/*
* file . c - NILFS regular file handling primitives including fsync ( ) .
*
* 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 Amagai Yoshiji < amagai @ osrg . net > ,
* Ryusuke Konishi < ryusuke @ osrg . net >
*/
# include <linux/fs.h>
# include <linux/mm.h>
# include <linux/writeback.h>
# include "nilfs.h"
# include "segment.h"
2011-07-17 04:44:56 +04:00
int nilfs_sync_file ( struct file * file , loff_t start , loff_t end , int datasync )
2009-04-07 06:01:33 +04:00
{
/*
* Called from fsync ( ) system call
* This is the only entry point that can catch write and synch
* timing for both data blocks and intermediate blocks .
*
* This function should be implemented when the writeback function
* will be implemented .
*/
2012-06-01 03:26:11 +04:00
struct the_nilfs * nilfs ;
2010-05-26 19:53:25 +04:00
struct inode * inode = file - > f_mapping - > host ;
2009-04-07 06:01:33 +04:00
int err ;
2011-07-17 04:44:56 +04:00
err = filemap_write_and_wait_range ( inode - > i_mapping , start , end ) ;
if ( err )
return err ;
mutex_lock ( & inode - > i_mutex ) ;
2012-06-01 03:26:11 +04:00
if ( nilfs_inode_dirty ( inode ) ) {
if ( datasync )
err = nilfs_construct_dsync_segment ( inode - > i_sb , inode ,
0 , LLONG_MAX ) ;
else
err = nilfs_construct_segment ( inode - > i_sb ) ;
2011-07-17 04:44:56 +04:00
}
mutex_unlock ( & inode - > i_mutex ) ;
2012-06-01 03:26:11 +04:00
nilfs = inode - > i_sb - > s_fs_info ;
if ( ! err & & nilfs_test_opt ( nilfs , BARRIER ) ) {
err = blkdev_issue_flush ( inode - > i_sb - > s_bdev , GFP_KERNEL , NULL ) ;
if ( err ! = - EIO )
err = 0 ;
}
2009-04-07 06:01:33 +04:00
return err ;
}
2009-04-07 06:01:37 +04:00
static int nilfs_page_mkwrite ( struct vm_area_struct * vma , struct vm_fault * vmf )
2009-04-07 06:01:33 +04:00
{
2009-04-07 06:01:37 +04:00
struct page * page = vmf - > page ;
struct inode * inode = vma - > vm_file - > f_dentry - > d_inode ;
struct nilfs_transaction_info ti ;
2012-06-12 18:20:44 +04:00
int ret = 0 ;
2009-04-07 06:01:37 +04:00
2011-03-09 05:05:08 +03:00
if ( unlikely ( nilfs_near_disk_full ( inode - > i_sb - > s_fs_info ) ) )
2009-04-07 06:01:37 +04:00
return VM_FAULT_SIGBUS ; /* -ENOSPC */
2012-06-12 18:20:44 +04:00
sb_start_pagefault ( inode - > i_sb ) ;
2009-04-07 06:01:37 +04:00
lock_page ( page ) ;
if ( page - > mapping ! = inode - > i_mapping | |
page_offset ( page ) > = i_size_read ( inode ) | | ! PageUptodate ( page ) ) {
unlock_page ( page ) ;
2012-06-12 18:20:44 +04:00
ret = - EFAULT ; /* make the VM retry the fault */
goto out ;
2009-04-07 06:01:37 +04:00
}
/*
* check to see if the page is mapped already ( no holes )
*/
2011-03-27 17:50:49 +04:00
if ( PageMappedToDisk ( page ) )
2009-04-07 06:01:37 +04:00
goto mapped ;
2011-03-27 17:50:49 +04:00
2009-04-07 06:01:37 +04:00
if ( page_has_buffers ( page ) ) {
struct buffer_head * bh , * head ;
int fully_mapped = 1 ;
bh = head = page_buffers ( page ) ;
do {
if ( ! buffer_mapped ( bh ) ) {
fully_mapped = 0 ;
break ;
}
} while ( bh = bh - > b_this_page , bh ! = head ) ;
if ( fully_mapped ) {
SetPageMappedToDisk ( page ) ;
goto mapped ;
}
}
unlock_page ( page ) ;
/*
* fill hole blocks
*/
ret = nilfs_transaction_begin ( inode - > i_sb , & ti , 1 ) ;
/* never returns -ENOMEM, but may return -ENOSPC */
if ( unlikely ( ret ) )
2012-06-12 18:20:44 +04:00
goto out ;
2009-04-07 06:01:37 +04:00
2012-10-01 07:04:56 +04:00
file_update_time ( vma - > vm_file ) ;
2012-06-12 18:20:44 +04:00
ret = __block_page_mkwrite ( vma , vmf , nilfs_get_block ) ;
if ( ret ) {
2009-04-07 06:01:37 +04:00
nilfs_transaction_abort ( inode - > i_sb ) ;
2012-06-12 18:20:44 +04:00
goto out ;
2009-04-07 06:01:37 +04:00
}
2011-03-27 17:50:49 +04:00
nilfs_set_file_dirty ( inode , 1 < < ( PAGE_SHIFT - inode - > i_blkbits ) ) ;
2009-04-07 06:01:37 +04:00
nilfs_transaction_commit ( inode - > i_sb ) ;
mapped :
2009-04-07 06:01:33 +04:00
wait_on_page_writeback ( page ) ;
2012-06-12 18:20:44 +04:00
out :
sb_end_pagefault ( inode - > i_sb ) ;
return block_page_mkwrite_return ( ret ) ;
2009-04-07 06:01:33 +04:00
}
2009-09-27 22:29:37 +04:00
static const struct vm_operations_struct nilfs_file_vm_ops = {
2009-04-07 06:01:33 +04:00
. fault = filemap_fault ,
. page_mkwrite = nilfs_page_mkwrite ,
} ;
static int nilfs_file_mmap ( struct file * file , struct vm_area_struct * vma )
{
file_accessed ( file ) ;
vma - > vm_ops = & nilfs_file_vm_ops ;
vma - > vm_flags | = VM_CAN_NONLINEAR ;
return 0 ;
}
/*
* We have mostly NULL ' s here : the current defaults are ok for
* the nilfs filesystem .
*/
2009-10-02 02:43:56 +04:00
const struct file_operations nilfs_file_operations = {
2009-04-07 06:01:33 +04:00
. llseek = generic_file_llseek ,
. read = do_sync_read ,
. write = do_sync_write ,
. aio_read = generic_file_aio_read ,
2009-04-07 06:01:38 +04:00
. aio_write = generic_file_aio_write ,
2009-04-07 06:01:53 +04:00
. unlocked_ioctl = nilfs_ioctl ,
2009-04-07 06:01:33 +04:00
# ifdef CONFIG_COMPAT
2011-02-03 15:26:17 +03:00
. compat_ioctl = nilfs_compat_ioctl ,
2009-04-07 06:01:33 +04:00
# endif /* CONFIG_COMPAT */
. mmap = nilfs_file_mmap ,
. open = generic_file_open ,
/* .release = nilfs_release_file, */
. fsync = nilfs_sync_file ,
. splice_read = generic_file_splice_read ,
} ;
2009-09-22 04:01:11 +04:00
const struct inode_operations nilfs_file_inode_operations = {
2009-04-07 06:01:33 +04:00
. truncate = nilfs_truncate ,
. setattr = nilfs_setattr ,
. permission = nilfs_permission ,
2010-12-26 10:38:43 +03:00
. fiemap = nilfs_fiemap ,
2009-04-07 06:01:33 +04:00
} ;
/* end of file */