2006-12-07 07:37:15 +03:00
/*
* Interface between ext4 and JBD
*/
2008-04-30 02:13:32 +04:00
# include "ext4_jbd2.h"
2006-12-07 07:37:15 +03:00
2009-11-23 04:52:12 +03:00
# include <trace/events/ext4.h>
2010-06-29 22:53:24 +04:00
int __ext4_journal_get_write_access ( const char * where , unsigned int line ,
handle_t * handle , struct buffer_head * bh )
2006-12-07 07:37:15 +03:00
{
2009-01-07 08:06:22 +03:00
int err = 0 ;
if ( ext4_handle_valid ( handle ) ) {
err = jbd2_journal_get_write_access ( handle , bh ) ;
if ( err )
2010-06-29 22:53:24 +04:00
ext4_journal_abort_handle ( where , line , __func__ , bh ,
2009-01-07 08:06:22 +03:00
handle , err ) ;
}
2006-12-07 07:37:15 +03:00
return err ;
}
2009-11-23 04:52:12 +03:00
/*
* The ext4 forget function must perform a revoke if we are freeing data
* which has been journaled . Metadata ( eg . indirect blocks ) must be
* revoked in all cases .
*
* " bh " may be NULL : a metadata block may have been freed from memory
* but there may still be a record of it in the journal , and that record
* still needs to be revoked .
*
* If the handle isn ' t valid we ' re not journaling , but we still need to
* call into ext4_journal_revoke ( ) to put the buffer head .
*/
2010-06-29 22:53:24 +04:00
int __ext4_forget ( const char * where , unsigned int line , handle_t * handle ,
int is_metadata , struct inode * inode ,
struct buffer_head * bh , ext4_fsblk_t blocknr )
2009-11-23 04:52:12 +03:00
{
int err ;
might_sleep ( ) ;
trace_ext4_forget ( inode , is_metadata , blocknr ) ;
BUFFER_TRACE ( bh , " enter " ) ;
jbd_debug ( 4 , " forgetting bh %p: is_metadata = %d, mode %o, "
" data mode %x \n " ,
bh , is_metadata , inode - > i_mode ,
test_opt ( inode - > i_sb , DATA_FLAGS ) ) ;
2009-11-24 19:05:59 +03:00
/* In the no journal case, we can just do a bforget and return */
if ( ! ext4_handle_valid ( handle ) ) {
bforget ( bh ) ;
return 0 ;
}
2009-11-23 04:52:12 +03:00
/* Never use the revoke function if we are doing full data
* journaling : there is no need to , and a V1 superblock won ' t
* support it . Otherwise , only skip the revoke on un - journaled
* data blocks . */
if ( test_opt ( inode - > i_sb , DATA_FLAGS ) = = EXT4_MOUNT_JOURNAL_DATA | |
( ! is_metadata & & ! ext4_should_journal_data ( inode ) ) ) {
if ( bh ) {
BUFFER_TRACE ( bh , " call jbd2_journal_forget " ) ;
2009-11-23 05:00:13 +03:00
err = jbd2_journal_forget ( handle , bh ) ;
if ( err )
2010-06-29 22:53:24 +04:00
ext4_journal_abort_handle ( where , line , __func__ ,
bh , handle , err ) ;
2009-11-23 05:00:13 +03:00
return err ;
2009-11-23 04:52:12 +03:00
}
return 0 ;
}
/*
* data ! = journal & & ( is_metadata | | should_journal_data ( inode ) )
*/
2009-11-24 19:05:59 +03:00
BUFFER_TRACE ( bh , " call jbd2_journal_revoke " ) ;
err = jbd2_journal_revoke ( handle , blocknr , bh ) ;
if ( err ) {
2010-06-29 22:53:24 +04:00
ext4_journal_abort_handle ( where , line , __func__ ,
bh , handle , err ) ;
2010-07-27 19:56:40 +04:00
__ext4_abort ( inode - > i_sb , where , line ,
" error %d when attempting revoke " , err ) ;
2009-11-24 19:05:59 +03:00
}
2009-11-23 04:52:12 +03:00
BUFFER_TRACE ( bh , " exit " ) ;
return err ;
}
2010-06-29 22:53:24 +04:00
int __ext4_journal_get_create_access ( const char * where , unsigned int line ,
2006-12-07 07:37:15 +03:00
handle_t * handle , struct buffer_head * bh )
{
2009-01-07 08:06:22 +03:00
int err = 0 ;
if ( ext4_handle_valid ( handle ) ) {
err = jbd2_journal_get_create_access ( handle , bh ) ;
if ( err )
2010-06-29 22:53:24 +04:00
ext4_journal_abort_handle ( where , line , __func__ ,
bh , handle , err ) ;
2009-01-07 08:06:22 +03:00
}
2006-12-07 07:37:15 +03:00
return err ;
}
2010-06-29 22:53:24 +04:00
int __ext4_handle_dirty_metadata ( const char * where , unsigned int line ,
handle_t * handle , struct inode * inode ,
struct buffer_head * bh )
2006-12-07 07:37:15 +03:00
{
2009-01-07 08:06:22 +03:00
int err = 0 ;
if ( ext4_handle_valid ( handle ) ) {
err = jbd2_journal_dirty_metadata ( handle , bh ) ;
2011-09-04 18:18:14 +04:00
if ( err ) {
/* Errors can only happen if there is a bug */
handle - > h_err = err ;
__ext4_journal_stop ( where , line , handle ) ;
}
2009-01-07 08:06:22 +03:00
} else {
2010-02-16 23:06:29 +03:00
if ( inode )
2009-09-12 21:41:55 +04:00
mark_buffer_dirty_inode ( bh , inode ) ;
else
mark_buffer_dirty ( bh ) ;
2009-01-07 08:06:22 +03:00
if ( inode & & inode_needs_sync ( inode ) ) {
sync_dirty_buffer ( bh ) ;
if ( buffer_req ( bh ) & & ! buffer_uptodate ( bh ) ) {
2010-07-27 19:56:03 +04:00
struct ext4_super_block * es ;
es = EXT4_SB ( inode - > i_sb ) - > s_es ;
es - > s_last_error_block =
cpu_to_le64 ( bh - > b_blocknr ) ;
2010-07-27 19:56:40 +04:00
ext4_error_inode ( inode , where , line ,
bh - > b_blocknr ,
" IO error syncing itable block " ) ;
2009-01-07 08:06:22 +03:00
err = - EIO ;
}
}
}
2006-12-07 07:37:15 +03:00
return err ;
}
2010-06-12 07:14:04 +04:00
2010-06-29 22:53:24 +04:00
int __ext4_handle_dirty_super ( const char * where , unsigned int line ,
handle_t * handle , struct super_block * sb )
2010-06-12 07:14:04 +04:00
{
struct buffer_head * bh = EXT4_SB ( sb ) - > s_sbh ;
int err = 0 ;
if ( ext4_handle_valid ( handle ) ) {
err = jbd2_journal_dirty_metadata ( handle , bh ) ;
if ( err )
2010-06-29 22:53:24 +04:00
ext4_journal_abort_handle ( where , line , __func__ ,
bh , handle , err ) ;
2010-06-12 07:14:04 +04:00
} else
sb - > s_dirt = 1 ;
return err ;
}