Re: [PATCH, RFC] ext4: Store basic fs error information in the superblock

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Theodore Ts'o <tytso@xxxxxxx> writes:

> Provide better RAS serviceability by stashing some basic file system
> error information in the ext4 superblock.
>
> This can help provide better hints as to what might have happened to a
> file system that contains errors when, for whatever reason,
> /var/log/messages or /var/log/kmsgs is not available.  Perhaps the
> hard drive got detached from computer containing the logs; perhaps the
> file system error was from long enough ago that the /var/log/messages
> information is no longer available or perhaps the user simply didn't
> bother to send them along.
Excellent choice. s_error_XXX fields finally allow one to detect errors
in determinant meaner. And most stress test finally will do it after
umount :).
Minor question: What snapshot staff is doing in that patch?
>
> Signed-off-by: "Theodore Ts'o" <tytso@xxxxxxx>
> ---
>  fs/ext4/block_validity.c |    8 ++-
>  fs/ext4/dir.c            |    8 ++-
>  fs/ext4/ext4.h           |   64 +++++++++++++++++--------
>  fs/ext4/ext4_jbd2.c      |   50 ++++++++++---------
>  fs/ext4/ext4_jbd2.h      |   47 ++++++++++---------
>  fs/ext4/extents.c        |   10 ++--
>  fs/ext4/inode.c          |   21 +++++---
>  fs/ext4/mballoc.c        |   15 +++---
>  fs/ext4/move_extent.c    |   10 ++--
>  fs/ext4/namei.c          |   14 +++--
>  fs/ext4/super.c          |  117 +++++++++++++++++++++++++++++----------------
>  11 files changed, 218 insertions(+), 146 deletions(-)
>
> diff --git a/fs/ext4/block_validity.c b/fs/ext4/block_validity.c
> index 5b6973f..d587dae 100644
> --- a/fs/ext4/block_validity.c
> +++ b/fs/ext4/block_validity.c
> @@ -229,16 +229,20 @@ int ext4_data_block_valid(struct ext4_sb_info *sbi, ext4_fsblk_t start_blk,
>  
>  	if ((start_blk <= le32_to_cpu(sbi->s_es->s_first_data_block)) ||
>  	    (start_blk + count < start_blk) ||
> -	    (start_blk + count > ext4_blocks_count(sbi->s_es)))
> +	    (start_blk + count > ext4_blocks_count(sbi->s_es))) {
> +		sbi->s_es->s_error_block = cpu_to_le64(start_blk);
>  		return 0;
> +	}
>  	while (n) {
>  		entry = rb_entry(n, struct ext4_system_zone, node);
>  		if (start_blk + count - 1 < entry->start_blk)
>  			n = n->rb_left;
>  		else if (start_blk >= (entry->start_blk + entry->count))
>  			n = n->rb_right;
> -		else
> +		else {
> +			sbi->s_es->s_error_block = cpu_to_le64(start_blk);
>  			return 0;
> +		}
>  	}
>  	return 1;
>  }
> diff --git a/fs/ext4/dir.c b/fs/ext4/dir.c
> index 2965c39..8c929ce 100644
> --- a/fs/ext4/dir.c
> +++ b/fs/ext4/dir.c
> @@ -61,7 +61,8 @@ static unsigned char get_dtype(struct super_block *sb, int filetype)
>  }
>  
>  
> -int ext4_check_dir_entry(const char *function, struct inode *dir,
> +int ext4_check_dir_entry(const char *function, unsigned int line,
> +			 struct inode *dir,
>  			 struct ext4_dir_entry_2 *de,
>  			 struct buffer_head *bh,
>  			 unsigned int offset)
> @@ -83,7 +84,7 @@ int ext4_check_dir_entry(const char *function, struct inode *dir,
>  		error_msg = "inode out of bounds";
>  
>  	if (error_msg != NULL)
> -		ext4_error_inode(function, dir,
> +		ext4_error_inode(function, line, dir,
>  			"bad entry in directory: %s - block=%llu"
>  			"offset=%u(%u), inode=%u, rec_len=%d, name_len=%d",
>  			error_msg, (unsigned long long) bh->b_blocknr,
> @@ -194,7 +195,8 @@ revalidate:
>  		while (!error && filp->f_pos < inode->i_size
>  		       && offset < sb->s_blocksize) {
>  			de = (struct ext4_dir_entry_2 *) (bh->b_data + offset);
> -			if (!ext4_check_dir_entry("ext4_readdir", inode, de,
> +			if (!ext4_check_dir_entry(__func__, __LINE__,
> +						  inode, de,
>  						  bh, offset)) {
>  				/*
>  				 * On error, skip the f_pos to the next block
> diff --git a/fs/ext4/ext4.h b/fs/ext4/ext4.h
> index 8b6d297..f51d961 100644
> --- a/fs/ext4/ext4.h
> +++ b/fs/ext4/ext4.h
> @@ -57,10 +57,10 @@
>  #endif
>  
>  #define EXT4_ERROR_INODE(inode, fmt, a...) \
> -	ext4_error_inode(__func__, (inode), (fmt), ## a)
> +	ext4_error_inode(__func__, __LINE__, (inode), (fmt), ## a)
>  
>  #define EXT4_ERROR_FILE(file, fmt, a...)	\
> -	ext4_error_file(__func__, (file), (fmt), ## a)
> +	ext4_error_file(__func__, __LINE__, (file), (fmt), ## a)
>  
>  /* data type for block offset of block group */
>  typedef int ext4_grpblk_t;
> @@ -1002,7 +1002,19 @@ struct ext4_super_block {
>  	__u8	s_reserved_char_pad2;
>  	__le16  s_reserved_pad;
>  	__le64	s_kbytes_written;	/* nr of lifetime kilobytes written */
> -	__u32   s_reserved[160];        /* Padding to the end of the block */
> +	__le32	s_snapshot_inum;	/* Inode number of active snapshot */
> +	__le32	s_snapshot_id;		/* sequential ID of active snapshot */
> +	__le64	s_snapshot_r_blocks_count; /* reserved blocks for active
> +					      snapshot's future use */
> +	__le32	s_snapshot_list;	/* inode number of the head of the on-disk snapshot list */
> +	__le32	s_first_error_time;	/* first time an error happened */
> +	__le32	s_last_error_time;	/* most recent time of an error */
> +	__le32	s_error_ino;		/* inode involved in last error */
> +	__le64	s_error_block;		/* block involved of last error */
> +	__le32	s_error_count;		/* number of fs errors */
> +	char	s_error_func[32];	/* function where the error happened */
> +	__le32	s_error_line;		/* line number where error happened */
> +	__le32   s_reserved[140];        /* Padding to the end of the block */
>  };
>  
>  #ifdef __KERNEL__
> @@ -1509,7 +1521,7 @@ extern unsigned ext4_init_block_bitmap(struct super_block *sb,
>  		ext4_init_block_bitmap(sb, NULL, group, desc)
>  
>  /* dir.c */
> -extern int ext4_check_dir_entry(const char *, struct inode *,
> +extern int ext4_check_dir_entry(const char *, unsigned int, struct inode *,
>  				struct ext4_dir_entry_2 *,
>  				struct buffer_head *, unsigned int);
>  extern int ext4_htree_store_dirent(struct file *dir_file, __u32 hash,
> @@ -1615,25 +1627,35 @@ extern int ext4_group_extend(struct super_block *sb,
>  				ext4_fsblk_t n_blocks_count);
>  
>  /* super.c */
> -extern void __ext4_error(struct super_block *, const char *, const char *, ...)
> -	__attribute__ ((format (printf, 3, 4)));
> -#define ext4_error(sb, message...)	__ext4_error(sb, __func__, ## message)
> -extern void ext4_error_inode(const char *, struct inode *, const char *, ...)
> -	__attribute__ ((format (printf, 3, 4)));
> -extern void ext4_error_file(const char *, struct file *, const char *, ...)
> -	__attribute__ ((format (printf, 3, 4)));
> -extern void __ext4_std_error(struct super_block *, const char *, int);
> -extern void ext4_abort(struct super_block *, const char *, const char *, ...)
> -	__attribute__ ((format (printf, 3, 4)));
> -extern void __ext4_warning(struct super_block *, const char *,
> +extern void __ext4_error(struct super_block *, const char *, unsigned int, const char *, ...)
> +	__attribute__ ((format (printf, 4, 5)));
> +#define ext4_error(sb, message...)	__ext4_error(sb, __func__, __LINE__, ## message)
> +extern void ext4_error_inode(const char *, unsigned int, struct inode *,
> +			     const char *, ...)
> +	__attribute__ ((format (printf, 4, 5)));
> +extern void ext4_error_file(const char *, unsigned int, struct file *,
> +			    const char *, ...)
> +	__attribute__ ((format (printf, 4, 5)));
> +extern void __ext4_std_error(struct super_block *, const char *,
> +			     unsigned int, int);
> +extern void __ext4_abort(struct super_block *, const char *, unsigned int,
> +		       const char *, ...)
> +	__attribute__ ((format (printf, 4, 5)));
> +#define ext4_abort(sb, message...)	__ext4_abort(sb, __func__, \
> +						       __LINE__, ## message)
> +extern void __ext4_warning(struct super_block *, const char *, unsigned int,
>  			  const char *, ...)
> -	__attribute__ ((format (printf, 3, 4)));
> -#define ext4_warning(sb, message...)	__ext4_warning(sb, __func__, ## message)
> +	__attribute__ ((format (printf, 4, 5)));
> +#define ext4_warning(sb, message...)	__ext4_warning(sb, __func__, \
> +						       __LINE__, ## message)
>  extern void ext4_msg(struct super_block *, const char *, const char *, ...)
>  	__attribute__ ((format (printf, 3, 4)));
> -extern void ext4_grp_locked_error(struct super_block *, ext4_group_t,
> -				const char *, const char *, ...)
> -	__attribute__ ((format (printf, 4, 5)));
> +extern void __ext4_grp_locked_error(const char *, unsigned int, \
> +				    struct super_block *, ext4_group_t, \
> +				    const char *, ...)
> +	__attribute__ ((format (printf, 5, 6)));
> +#define ext4_grp_locked_error(sb, grp, message...) \
> +	__ext4_grp_locked_error(__func__, __LINE__, (sb), (grp), ## message)
>  extern void ext4_update_dynamic_rev(struct super_block *sb);
>  extern int ext4_update_compat_feature(handle_t *handle, struct super_block *sb,
>  					__u32 compat);
> @@ -1767,7 +1789,7 @@ static inline unsigned int ext4_flex_bg_size(struct ext4_sb_info *sbi)
>  #define ext4_std_error(sb, errno)				\
>  do {								\
>  	if ((errno))						\
> -		__ext4_std_error((sb), __func__, (errno));	\
> +		__ext4_std_error((sb), __func__, __LINE__, (errno));	\
>  } while (0)
>  
>  #ifdef CONFIG_SMP
> diff --git a/fs/ext4/ext4_jbd2.c b/fs/ext4/ext4_jbd2.c
> index cfd27b3..93b83e8 100644
> --- a/fs/ext4/ext4_jbd2.c
> +++ b/fs/ext4/ext4_jbd2.c
> @@ -6,29 +6,29 @@
>  
>  #include <trace/events/ext4.h>
>  
> -int __ext4_journal_get_undo_access(const char *where, handle_t *handle,
> -				struct buffer_head *bh)
> +int __ext4_journal_get_undo_access(const char *where, unsigned int line,
> +				   handle_t *handle, struct buffer_head *bh)
>  {
>  	int err = 0;
>  
>  	if (ext4_handle_valid(handle)) {
>  		err = jbd2_journal_get_undo_access(handle, bh);
>  		if (err)
> -			ext4_journal_abort_handle(where, __func__, bh,
> +			ext4_journal_abort_handle(where, line, __func__, bh,
>  						  handle, err);
>  	}
>  	return err;
>  }
>  
> -int __ext4_journal_get_write_access(const char *where, handle_t *handle,
> -				struct buffer_head *bh)
> +int __ext4_journal_get_write_access(const char *where, unsigned int line,
> +				    handle_t *handle, struct buffer_head *bh)
>  {
>  	int err = 0;
>  
>  	if (ext4_handle_valid(handle)) {
>  		err = jbd2_journal_get_write_access(handle, bh);
>  		if (err)
> -			ext4_journal_abort_handle(where, __func__, bh,
> +			ext4_journal_abort_handle(where, line, __func__, bh,
>  						  handle, err);
>  	}
>  	return err;
> @@ -46,9 +46,9 @@ int __ext4_journal_get_write_access(const char *where, handle_t *handle,
>   * 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.
>   */
> -int __ext4_forget(const char *where, handle_t *handle, int is_metadata,
> -		  struct inode *inode, struct buffer_head *bh,
> -		  ext4_fsblk_t blocknr)
> +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)
>  {
>  	int err;
>  
> @@ -79,8 +79,8 @@ int __ext4_forget(const char *where, handle_t *handle, int is_metadata,
>  			BUFFER_TRACE(bh, "call jbd2_journal_forget");
>  			err = jbd2_journal_forget(handle, bh);
>  			if (err)
> -				ext4_journal_abort_handle(where, __func__, bh,
> -							  handle, err);
> +				ext4_journal_abort_handle(where, line, __func__,
> +							  bh, handle, err);
>  			return err;
>  		}
>  		return 0;
> @@ -92,15 +92,16 @@ int __ext4_forget(const char *where, handle_t *handle, int is_metadata,
>  	BUFFER_TRACE(bh, "call jbd2_journal_revoke");
>  	err = jbd2_journal_revoke(handle, blocknr, bh);
>  	if (err) {
> -		ext4_journal_abort_handle(where, __func__, bh, handle, err);
> -		ext4_abort(inode->i_sb, __func__,
> +		ext4_journal_abort_handle(where, line, __func__, 
> +					  bh, handle, err);
> +		__ext4_abort(inode->i_sb, where, line,
>  			   "error %d when attempting revoke", err);
>  	}
>  	BUFFER_TRACE(bh, "exit");
>  	return err;
>  }
>  
> -int __ext4_journal_get_create_access(const char *where,
> +int __ext4_journal_get_create_access(const char *where, unsigned int line,
>  				handle_t *handle, struct buffer_head *bh)
>  {
>  	int err = 0;
> @@ -108,22 +109,23 @@ int __ext4_journal_get_create_access(const char *where,
>  	if (ext4_handle_valid(handle)) {
>  		err = jbd2_journal_get_create_access(handle, bh);
>  		if (err)
> -			ext4_journal_abort_handle(where, __func__, bh,
> -						  handle, err);
> +			ext4_journal_abort_handle(where, line, __func__,
> +						  bh, handle, err);
>  	}
>  	return err;
>  }
>  
> -int __ext4_handle_dirty_metadata(const char *where, handle_t *handle,
> -				 struct inode *inode, struct buffer_head *bh)
> +int __ext4_handle_dirty_metadata(const char *where, unsigned int line,
> +				 handle_t *handle, struct inode *inode,
> +				 struct buffer_head *bh)
>  {
>  	int err = 0;
>  
>  	if (ext4_handle_valid(handle)) {
>  		err = jbd2_journal_dirty_metadata(handle, bh);
>  		if (err)
> -			ext4_journal_abort_handle(where, __func__, bh,
> -						  handle, err);
> +			ext4_journal_abort_handle(where, line,__func__,
> +						  bh, handle, err);
>  	} else {
>  		if (inode)
>  			mark_buffer_dirty_inode(bh, inode);
> @@ -144,8 +146,8 @@ int __ext4_handle_dirty_metadata(const char *where, handle_t *handle,
>  	return err;
>  }
>  
> -int __ext4_handle_dirty_super(const char *where, handle_t *handle,
> -			      struct super_block *sb)
> +int __ext4_handle_dirty_super(const char *where, unsigned int line,
> +			      handle_t *handle, struct super_block *sb)
>  {
>  	struct buffer_head *bh = EXT4_SB(sb)->s_sbh;
>  	int err = 0;
> @@ -153,8 +155,8 @@ int __ext4_handle_dirty_super(const char *where, handle_t *handle,
>  	if (ext4_handle_valid(handle)) {
>  		err = jbd2_journal_dirty_metadata(handle, bh);
>  		if (err)
> -			ext4_journal_abort_handle(where, __func__, bh,
> -						  handle, err);
> +			ext4_journal_abort_handle(where, line, __func__, 
> +						  bh, handle, err);
>  	} else
>  		sb->s_dirt = 1;
>  	return err;
> diff --git a/fs/ext4/ext4_jbd2.h b/fs/ext4/ext4_jbd2.h
> index 38d1e66..b0bd792 100644
> --- a/fs/ext4/ext4_jbd2.h
> +++ b/fs/ext4/ext4_jbd2.h
> @@ -122,44 +122,47 @@ int ext4_mark_inode_dirty(handle_t *handle, struct inode *inode);
>  /*
>   * Wrapper functions with which ext4 calls into JBD.
>   */
> -void ext4_journal_abort_handle(const char *caller, const char *err_fn,
> +void ext4_journal_abort_handle(const char *caller, unsigned int line,
> +			       const char *err_fn,
>  		struct buffer_head *bh, handle_t *handle, int err);
>  
> -int __ext4_journal_get_undo_access(const char *where, handle_t *handle,
> -				struct buffer_head *bh);
> +int __ext4_journal_get_undo_access(const char *where, unsigned int line,
> +				   handle_t *handle, struct buffer_head *bh);
>  
> -int __ext4_journal_get_write_access(const char *where, handle_t *handle,
> -				struct buffer_head *bh);
> +int __ext4_journal_get_write_access(const char *where, unsigned int line,
> +				    handle_t *handle, struct buffer_head *bh);
>  
> -int __ext4_forget(const char *where, handle_t *handle, int is_metadata,
> -		  struct inode *inode, struct buffer_head *bh,
> -		  ext4_fsblk_t blocknr);
> +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);
>  
> -int __ext4_journal_get_create_access(const char *where,
> +int __ext4_journal_get_create_access(const char *where, unsigned int line,
>  				handle_t *handle, struct buffer_head *bh);
>  
> -int __ext4_handle_dirty_metadata(const char *where, handle_t *handle,
> -				 struct inode *inode, struct buffer_head *bh);
> +int __ext4_handle_dirty_metadata(const char *where, unsigned int line,
> +				 handle_t *handle, struct inode *inode,
> +				 struct buffer_head *bh);
>  
> -int __ext4_handle_dirty_super(const char *where, handle_t *handle,
> -			      struct super_block *sb);
> +int __ext4_handle_dirty_super(const char *where, unsigned int line,
> +			      handle_t *handle, struct super_block *sb);
>  
>  #define ext4_journal_get_undo_access(handle, bh) \
> -	__ext4_journal_get_undo_access(__func__, (handle), (bh))
> +	__ext4_journal_get_undo_access(__func__, __LINE__, (handle), (bh))
>  #define ext4_journal_get_write_access(handle, bh) \
> -	__ext4_journal_get_write_access(__func__, (handle), (bh))
> +	__ext4_journal_get_write_access(__func__, __LINE__, (handle), (bh))
>  #define ext4_forget(handle, is_metadata, inode, bh, block_nr) \
> -	__ext4_forget(__func__, (handle), (is_metadata), (inode), (bh),\
> -		      (block_nr))
> +	__ext4_forget(__func__, __LINE__, (handle), (is_metadata), (inode), \
> +		      (bh), (block_nr))
>  #define ext4_journal_get_create_access(handle, bh) \
> -	__ext4_journal_get_create_access(__func__, (handle), (bh))
> +	__ext4_journal_get_create_access(__func__, __LINE__, (handle), (bh))
>  #define ext4_handle_dirty_metadata(handle, inode, bh) \
> -	__ext4_handle_dirty_metadata(__func__, (handle), (inode), (bh))
> +	__ext4_handle_dirty_metadata(__func__, __LINE__, (handle), (inode), \
> +				     (bh))
>  #define ext4_handle_dirty_super(handle, sb) \
> -	__ext4_handle_dirty_super(__func__, (handle), (sb))
> +	__ext4_handle_dirty_super(__func__, __LINE__, (handle), (sb))
>  
>  handle_t *ext4_journal_start_sb(struct super_block *sb, int nblocks);
> -int __ext4_journal_stop(const char *where, handle_t *handle);
> +int __ext4_journal_stop(const char *where, unsigned int line, handle_t *handle);
>  
>  #define EXT4_NOJOURNAL_MAX_REF_COUNT ((unsigned long) 4096)
>  
> @@ -212,7 +215,7 @@ static inline handle_t *ext4_journal_start(struct inode *inode, int nblocks)
>  }
>  
>  #define ext4_journal_stop(handle) \
> -	__ext4_journal_stop(__func__, (handle))
> +	__ext4_journal_stop(__func__, __LINE__, (handle))
>  
>  static inline handle_t *ext4_journal_current_handle(void)
>  {
> diff --git a/fs/ext4/extents.c b/fs/ext4/extents.c
> index 346de3d..fac5516 100644
> --- a/fs/ext4/extents.c
> +++ b/fs/ext4/extents.c
> @@ -401,9 +401,9 @@ static int ext4_valid_extent_entries(struct inode *inode,
>  	return 1;
>  }
>  
> -static int __ext4_ext_check(const char *function, struct inode *inode,
> -					struct ext4_extent_header *eh,
> -					int depth)
> +static int __ext4_ext_check(const char *function, unsigned int line,
> +			    struct inode *inode, struct ext4_extent_header *eh,
> +			    int depth)
>  {
>  	const char *error_msg;
>  	int max = 0;
> @@ -436,7 +436,7 @@ static int __ext4_ext_check(const char *function, struct inode *inode,
>  	return 0;
>  
>  corrupted:
> -	ext4_error_inode(function, inode,
> +	ext4_error_inode(function, line, inode,
>  			"bad header/extent: %s - magic %x, "
>  			"entries %u, max %u(%u), depth %u(%u)",
>  			error_msg, le16_to_cpu(eh->eh_magic),
> @@ -447,7 +447,7 @@ corrupted:
>  }
>  
>  #define ext4_ext_check(inode, eh, depth)	\
> -	__ext4_ext_check(__func__, inode, eh, depth)
> +	__ext4_ext_check(__func__, __LINE__, inode, eh, depth)
>  
>  int ext4_ext_check_inode(struct inode *inode)
>  {
> diff --git a/fs/ext4/inode.c b/fs/ext4/inode.c
> index b485987..aade5e4 100644
> --- a/fs/ext4/inode.c
> +++ b/fs/ext4/inode.c
> @@ -337,9 +337,11 @@ static int ext4_block_to_path(struct inode *inode,
>  	return n;
>  }
>  
> -static int __ext4_check_blockref(const char *function, struct inode *inode,
> +static int __ext4_check_blockref(const char *function, unsigned int line,
> +				 struct inode *inode,
>  				 __le32 *p, unsigned int max)
>  {
> +	struct ext4_super_block *es = EXT4_SB(inode->i_sb)->s_es;
>  	__le32 *bref = p;
>  	unsigned int blk;
>  
> @@ -348,7 +350,8 @@ static int __ext4_check_blockref(const char *function, struct inode *inode,
>  		if (blk &&
>  		    unlikely(!ext4_data_block_valid(EXT4_SB(inode->i_sb),
>  						    blk, 1))) {
> -			ext4_error_inode(function, inode,
> +			es->s_error_block = cpu_to_le64(blk);
> +			ext4_error_inode(function, line, inode,
>  					 "invalid block reference %u", blk);
>  			return -EIO;
>  		}
> @@ -358,11 +361,13 @@ static int __ext4_check_blockref(const char *function, struct inode *inode,
>  
>  
>  #define ext4_check_indirect_blockref(inode, bh)                         \
> -	__ext4_check_blockref(__func__, inode, (__le32 *)(bh)->b_data,  \
> +	__ext4_check_blockref(__func__, __LINE__, inode,		\
> +			      (__le32 *)(bh)->b_data,			\
>  			      EXT4_ADDR_PER_BLOCK((inode)->i_sb))
>  
>  #define ext4_check_inode_blockref(inode)                                \
> -	__ext4_check_blockref(__func__, inode, EXT4_I(inode)->i_data,   \
> +	__ext4_check_blockref(__func__, __LINE__, inode,		\
> +			      EXT4_I(inode)->i_data,			\
>  			      EXT4_NDIR_BLOCKS)
>  
>  /**
> @@ -1129,11 +1134,12 @@ void ext4_da_update_reserve_space(struct inode *inode,
>  }
>  
>  static int check_block_validity(struct inode *inode, const char *func,
> +				unsigned int line,
>  				struct ext4_map_blocks *map)
>  {
>  	if (!ext4_data_block_valid(EXT4_SB(inode->i_sb), map->m_pblk,
>  				   map->m_len)) {
> -		ext4_error_inode(func, inode,
> +		ext4_error_inode(func, line, inode,
>  			   "lblock %lu mapped to illegal pblock %llu "
>  			   "(length %d)", (unsigned long) map->m_lblk,
>  				 map->m_pblk, map->m_len);
> @@ -1244,7 +1250,7 @@ int ext4_map_blocks(handle_t *handle, struct inode *inode,
>  	up_read((&EXT4_I(inode)->i_data_sem));
>  
>  	if (retval > 0 && map->m_flags & EXT4_MAP_MAPPED) {
> -		int ret = check_block_validity(inode, __func__, map);
> +		int ret = check_block_validity(inode, __func__, __LINE__, map);
>  		if (ret != 0)
>  			return ret;
>  	}
> @@ -1324,8 +1330,7 @@ int ext4_map_blocks(handle_t *handle, struct inode *inode,
>  
>  	up_write((&EXT4_I(inode)->i_data_sem));
>  	if (retval > 0 && map->m_flags & EXT4_MAP_MAPPED) {
> -		int ret = check_block_validity(inode,
> -					       "ext4_map_blocks_after_alloc",
> +		int ret = check_block_validity(inode, __func__, __LINE__,
>  					       map);
>  		if (ret != 0)
>  			return ret;
> diff --git a/fs/ext4/mballoc.c b/fs/ext4/mballoc.c
> index b2948b0..92ae5e8 100644
> --- a/fs/ext4/mballoc.c
> +++ b/fs/ext4/mballoc.c
> @@ -446,7 +446,7 @@ static void mb_free_blocks_double(struct inode *inode, struct ext4_buddy *e4b,
>  			blocknr = ext4_group_first_block_no(sb, e4b->bd_group);
>  			blocknr += first + i;
>  			ext4_grp_locked_error(sb, e4b->bd_group,
> -				   __func__, "double-free of inode"
> +				   "double-free of inode"
>  				   " %lu's block %llu(bit %u in group %u)",
>  				   inode ? inode->i_ino : 0, blocknr,
>  				   first + i, e4b->bd_group);
> @@ -712,7 +712,7 @@ void ext4_mb_generate_buddy(struct super_block *sb,
>  	grp->bb_fragments = fragments;
>  
>  	if (free != grp->bb_free) {
> -		ext4_grp_locked_error(sb, group,  __func__,
> +		ext4_grp_locked_error(sb, group,
>  			"EXT4-fs: group %u: %u blocks in bitmap, %u in gd",
>  			group, free, grp->bb_free);
>  		/*
> @@ -1296,7 +1296,7 @@ static void mb_free_blocks(struct inode *inode, struct ext4_buddy *e4b,
>  			blocknr = ext4_group_first_block_no(sb, e4b->bd_group);
>  			blocknr += block;
>  			ext4_grp_locked_error(sb, e4b->bd_group,
> -				   __func__, "double-free of inode"
> +				   "double-free of inode"
>  				   " %lu's block %llu(bit %u in group %u)",
>  				   inode ? inode->i_ino : 0, blocknr, block,
>  				   e4b->bd_group);
> @@ -1789,7 +1789,7 @@ void ext4_mb_complex_scan_group(struct ext4_allocation_context *ac,
>  			 * we have free blocks
>  			 */
>  			ext4_grp_locked_error(sb, e4b->bd_group,
> -					__func__, "%d free blocks as per "
> +					"%d free blocks as per "
>  					"group info. But bitmap says 0",
>  					free);
>  			break;
> @@ -1799,7 +1799,7 @@ void ext4_mb_complex_scan_group(struct ext4_allocation_context *ac,
>  		BUG_ON(ex.fe_len <= 0);
>  		if (free < ex.fe_len) {
>  			ext4_grp_locked_error(sb, e4b->bd_group,
> -					__func__, "%d free blocks as per "
> +					"%d free blocks as per "
>  					"group info. But got %d blocks",
>  					free, ex.fe_len);
>  			/*
> @@ -3584,8 +3584,7 @@ ext4_mb_release_inode_pa(struct ext4_buddy *e4b, struct buffer_head *bitmap_bh,
>  			pa, (unsigned long) pa->pa_lstart,
>  			(unsigned long) pa->pa_pstart,
>  			(unsigned long) pa->pa_len);
> -		ext4_grp_locked_error(sb, group,
> -					__func__, "free %u, pa_free %u",
> +		ext4_grp_locked_error(sb, group, "free %u, pa_free %u",
>  					free, pa->pa_free);
>  		/*
>  		 * pa is already deleted so we use the value obtained
> @@ -4427,7 +4426,7 @@ ext4_mb_free_metadata(handle_t *handle, struct ext4_buddy *e4b,
>  		else if (block >= (entry->start_blk + entry->count))
>  			n = &(*n)->rb_right;
>  		else {
> -			ext4_grp_locked_error(sb, e4b->bd_group, __func__,
> +			ext4_grp_locked_error(sb, e4b->bd_group,
>  					"Double free of blocks %d (%d %d)",
>  					block, entry->start_blk, entry->count);
>  			return 0;
> diff --git a/fs/ext4/move_extent.c b/fs/ext4/move_extent.c
> index 52abfa1..5f1ed9f 100644
> --- a/fs/ext4/move_extent.c
> +++ b/fs/ext4/move_extent.c
> @@ -148,17 +148,17 @@ mext_next_extent(struct inode *inode, struct ext4_ext_path *path,
>   */
>  static int
>  mext_check_null_inode(struct inode *inode1, struct inode *inode2,
> -		const char *function)
> +		      const char *function, unsigned int line)
>  {
>  	int ret = 0;
>  
>  	if (inode1 == NULL) {
> -		__ext4_error(inode2->i_sb, function,
> +		__ext4_error(inode2->i_sb, function, line,
>  			"Both inodes should not be NULL: "
>  			"inode1 NULL inode2 %lu", inode2->i_ino);
>  		ret = -EIO;
>  	} else if (inode2 == NULL) {
> -		__ext4_error(inode1->i_sb, function,
> +		__ext4_error(inode1->i_sb, function, line,
>  			"Both inodes should not be NULL: "
>  			"inode1 %lu inode2 NULL", inode1->i_ino);
>  		ret = -EIO;
> @@ -1084,7 +1084,7 @@ mext_inode_double_lock(struct inode *inode1, struct inode *inode2)
>  
>  	BUG_ON(inode1 == NULL && inode2 == NULL);
>  
> -	ret = mext_check_null_inode(inode1, inode2, __func__);
> +	ret = mext_check_null_inode(inode1, inode2, __func__, __LINE__);
>  	if (ret < 0)
>  		goto out;
>  
> @@ -1121,7 +1121,7 @@ mext_inode_double_unlock(struct inode *inode1, struct inode *inode2)
>  
>  	BUG_ON(inode1 == NULL && inode2 == NULL);
>  
> -	ret = mext_check_null_inode(inode1, inode2, __func__);
> +	ret = mext_check_null_inode(inode1, inode2, __func__, __LINE__);
>  	if (ret < 0)
>  		goto out;
>  
> diff --git a/fs/ext4/namei.c b/fs/ext4/namei.c
> index 5a61f77..66b6e1c 100644
> --- a/fs/ext4/namei.c
> +++ b/fs/ext4/namei.c
> @@ -605,7 +605,7 @@ static int htree_dirblock_to_tree(struct file *dir_file,
>  					   dir->i_sb->s_blocksize -
>  					   EXT4_DIR_REC_LEN(0));
>  	for (; de < top; de = ext4_next_entry(de, dir->i_sb->s_blocksize)) {
> -		if (!ext4_check_dir_entry("htree_dirblock_to_tree", dir, de, bh,
> +		if (!ext4_check_dir_entry(__func__, __LINE__, dir, de, bh,
>  					(block<<EXT4_BLOCK_SIZE_BITS(dir->i_sb))
>  						+((char *)de - bh->b_data))) {
>  			/* On error, skip the f_pos to the next block. */
> @@ -844,7 +844,7 @@ static inline int search_dirblock(struct buffer_head *bh,
>  		if ((char *) de + namelen <= dlimit &&
>  		    ext4_match (namelen, name, de)) {
>  			/* found a match - just to be sure, do a full check */
> -			if (!ext4_check_dir_entry("ext4_find_entry",
> +			if (!ext4_check_dir_entry(__func__, __LINE__,
>  						  dir, de, bh, offset))
>  				return -1;
>  			*res_dir = de;
> @@ -1019,7 +1019,8 @@ static struct buffer_head * ext4_dx_find_entry(struct inode *dir, const struct q
>  			int off = (block << EXT4_BLOCK_SIZE_BITS(sb))
>  				  + ((char *) de - bh->b_data);
>  
> -			if (!ext4_check_dir_entry(__func__, dir, de, bh, off)) {
> +			if (!ext4_check_dir_entry(__func__, __LINE__, dir,
> +						  de, bh, off)) {
>  				brelse(bh);
>  				*err = ERR_BAD_DX_DIR;
>  				goto errout;
> @@ -1303,7 +1304,7 @@ static int add_dirent_to_buf(handle_t *handle, struct dentry *dentry,
>  		de = (struct ext4_dir_entry_2 *)bh->b_data;
>  		top = bh->b_data + blocksize - reclen;
>  		while ((char *) de <= top) {
> -			if (!ext4_check_dir_entry("ext4_add_entry", dir, de,
> +			if (!ext4_check_dir_entry(__func__, __LINE__, dir, de,
>  						  bh, offset))
>  				return -EIO;
>  			if (ext4_match(namelen, name, de))
> @@ -1671,7 +1672,7 @@ static int ext4_delete_entry(handle_t *handle,
>  	pde = NULL;
>  	de = (struct ext4_dir_entry_2 *) bh->b_data;
>  	while (i < bh->b_size) {
> -		if (!ext4_check_dir_entry("ext4_delete_entry", dir, de, bh, i))
> +		if (!ext4_check_dir_entry(__func__, __LINE__, dir, de, bh, i))
>  			return -EIO;
>  		if (de == de_del)  {
>  			BUFFER_TRACE(bh, "get_write_access");
> @@ -1954,7 +1955,8 @@ static int empty_dir(struct inode *inode)
>  			}
>  			de = (struct ext4_dir_entry_2 *) bh->b_data;
>  		}
> -		if (!ext4_check_dir_entry("empty_dir", inode, de, bh, offset)) {
> +		if (!ext4_check_dir_entry(__func__, __LINE__, inode,
> +					  de, bh, offset)) {
>  			de = (struct ext4_dir_entry_2 *)(bh->b_data +
>  							 sb->s_blocksize);
>  			offset = (offset | (sb->s_blocksize - 1)) + 1;
> diff --git a/fs/ext4/super.c b/fs/ext4/super.c
> index 422a4ce..0c689fb 100644
> --- a/fs/ext4/super.c
> +++ b/fs/ext4/super.c
> @@ -248,7 +248,7 @@ handle_t *ext4_journal_start_sb(struct super_block *sb, int nblocks)
>  	journal = EXT4_SB(sb)->s_journal;
>  	if (journal) {
>  		if (is_journal_aborted(journal)) {
> -			ext4_abort(sb, __func__, "Detected aborted journal");
> +			ext4_abort(sb, "Detected aborted journal");
>  			return ERR_PTR(-EROFS);
>  		}
>  		return jbd2_journal_start(journal, nblocks);
> @@ -262,7 +262,7 @@ handle_t *ext4_journal_start_sb(struct super_block *sb, int nblocks)
>   * that sync() will call the filesystem's write_super callback if
>   * appropriate.
>   */
> -int __ext4_journal_stop(const char *where, handle_t *handle)
> +int __ext4_journal_stop(const char *where, unsigned int line, handle_t *handle)
>  {
>  	struct super_block *sb;
>  	int err;
> @@ -279,12 +279,13 @@ int __ext4_journal_stop(const char *where, handle_t *handle)
>  	if (!err)
>  		err = rc;
>  	if (err)
> -		__ext4_std_error(sb, where, err);
> +		__ext4_std_error(sb, where, line, err);
>  	return err;
>  }
>  
> -void ext4_journal_abort_handle(const char *caller, const char *err_fn,
> -		struct buffer_head *bh, handle_t *handle, int err)
> +void ext4_journal_abort_handle(const char *caller, unsigned int line,
> +			       const char *err_fn, struct buffer_head *bh,
> +			       handle_t *handle, int err)
>  {
>  	char nbuf[16];
>  	const char *errstr = ext4_decode_error(NULL, err, nbuf);
> @@ -300,8 +301,8 @@ void ext4_journal_abort_handle(const char *caller, const char *err_fn,
>  	if (is_handle_aborted(handle))
>  		return;
>  
> -	printk(KERN_ERR "%s: aborting transaction: %s in %s\n",
> -	       caller, errstr, err_fn);
> +	printk(KERN_ERR "%s:%d: aborting transaction: %s in %s\n",
> +	       caller, line, errstr, err_fn);
>  
>  	jbd2_journal_abort_handle(handle);
>  }
> @@ -327,6 +328,10 @@ static void ext4_handle_error(struct super_block *sb)
>  
>  	EXT4_SB(sb)->s_mount_state |= EXT4_ERROR_FS;
>  	es->s_state |= cpu_to_le16(EXT4_ERROR_FS);
> +	es->s_last_error_time = cpu_to_le32(get_seconds());
> +	if (!es->s_first_error_time)
> +		es->s_first_error_time = es->s_last_error_time;
> +	es->s_error_count = cpu_to_le32(le32_to_cpu(es->s_error_count) + 1);
>  
>  	if (sb->s_flags & MS_RDONLY)
>  		return;
> @@ -349,12 +354,16 @@ static void ext4_handle_error(struct super_block *sb)
>  }
>  
>  void __ext4_error(struct super_block *sb, const char *function,
> -		const char *fmt, ...)
> +		  unsigned int line, const char *fmt, ...)
>  {
>  	va_list args;
> +	struct ext4_super_block *es = EXT4_SB(sb)->s_es;
>  
> +	strncpy(es->s_error_func, function, sizeof(es->s_error_func));
> +	es->s_error_line = cpu_to_le32(line);
>  	va_start(args, fmt);
> -	printk(KERN_CRIT "EXT4-fs error (device %s): %s: ", sb->s_id, function);
> +	printk(KERN_CRIT "EXT4-fs error (device %s): %s:%d: ", sb->s_id,
> +	       function, line);
>  	vprintk(fmt, args);
>  	printk("\n");
>  	va_end(args);
> @@ -362,14 +371,18 @@ void __ext4_error(struct super_block *sb, const char *function,
>  	ext4_handle_error(sb);
>  }
>  
> -void ext4_error_inode(const char *function, struct inode *inode,
> -		      const char *fmt, ...)
> +void ext4_error_inode(const char *function, unsigned int line,
> +		      struct inode *inode, const char *fmt, ...)
>  {
>  	va_list args;
> +	struct ext4_super_block *es = EXT4_SB(inode->i_sb)->s_es;
>  
> +	strncpy(es->s_error_func, function, sizeof(es->s_error_func));
> +	es->s_error_line = cpu_to_le32(line);
>  	va_start(args, fmt);
> -	printk(KERN_CRIT "EXT4-fs error (device %s): %s: inode #%lu: (comm %s) ",
> -	       inode->i_sb->s_id, function, inode->i_ino, current->comm);
> +	printk(KERN_CRIT "EXT4-fs error (device %s): %s:%d: inode #%lu: "
> +	       "(comm %s) ",
> +	       inode->i_sb->s_id, function, line, inode->i_ino, current->comm);
>  	vprintk(fmt, args);
>  	printk("\n");
>  	va_end(args);
> @@ -377,20 +390,26 @@ void ext4_error_inode(const char *function, struct inode *inode,
>  	ext4_handle_error(inode->i_sb);
>  }
>  
> -void ext4_error_file(const char *function, struct file *file,
> -		     const char *fmt, ...)
> +void ext4_error_file(const char *function, unsigned int line,
> +		     struct file *file, const char *fmt, ...)
>  {
>  	va_list args;
> +	struct ext4_super_block *es;
>  	struct inode *inode = file->f_dentry->d_inode;
>  	char pathname[80], *path;
>  
> +	es = EXT4_SB(inode->i_sb)->s_es;
> +	strncpy(es->s_error_func, function, sizeof(es->s_error_func));
> +	es->s_error_line = cpu_to_le32(line);
>  	va_start(args, fmt);
>  	path = d_path(&(file->f_path), pathname, sizeof(pathname));
>  	if (!path)
>  		path = "(unknown)";
>  	printk(KERN_CRIT
> -	       "EXT4-fs error (device %s): %s: inode #%lu (comm %s path %s): ",
> -	       inode->i_sb->s_id, function, inode->i_ino, current->comm, path);
> +	       "EXT4-fs error (device %s): %s:%d: inode #%lu "
> +	       "(comm %s path %s): ",
> +	       inode->i_sb->s_id, function, line, inode->i_ino,
> +	       current->comm, path);
>  	vprintk(fmt, args);
>  	printk("\n");
>  	va_end(args);
> @@ -435,10 +454,12 @@ static const char *ext4_decode_error(struct super_block *sb, int errno,
>  /* __ext4_std_error decodes expected errors from journaling functions
>   * automatically and invokes the appropriate error response.  */
>  
> -void __ext4_std_error(struct super_block *sb, const char *function, int errno)
> +void __ext4_std_error(struct super_block *sb, const char *function,
> +		      unsigned int line, int errno)
>  {
>  	char nbuf[16];
>  	const char *errstr;
> +	struct ext4_super_block *es = EXT4_SB(sb)->s_es;
>  
>  	/* Special case: if the error is EROFS, and we're not already
>  	 * inside a transaction, then there's really no point in logging
> @@ -448,8 +469,10 @@ void __ext4_std_error(struct super_block *sb, const char *function, int errno)
>  		return;
>  
>  	errstr = ext4_decode_error(sb, errno, nbuf);
> -	printk(KERN_CRIT "EXT4-fs error (device %s) in %s: %s\n",
> -	       sb->s_id, function, errstr);
> +	printk(KERN_CRIT "EXT4-fs error (device %s) in %s:%d: %s\n",
> +	       sb->s_id, function, line, errstr);
> +	strncpy(es->s_error_func, function, sizeof(es->s_error_func));
> +	es->s_error_line = cpu_to_le32(line);
>  
>  	ext4_handle_error(sb);
>  }
> @@ -464,29 +487,32 @@ void __ext4_std_error(struct super_block *sb, const char *function, int errno)
>   * case we take the easy way out and panic immediately.
>   */
>  
> -void ext4_abort(struct super_block *sb, const char *function,
> -		const char *fmt, ...)
> +void __ext4_abort(struct super_block *sb, const char *function,
> +		unsigned int line, const char *fmt, ...)
>  {
>  	va_list args;
> +	struct ext4_super_block *es = EXT4_SB(sb)->s_es;
>  
>  	va_start(args, fmt);
> -	printk(KERN_CRIT "EXT4-fs error (device %s): %s: ", sb->s_id, function);
> +	printk(KERN_CRIT "EXT4-fs error (device %s): %s:%d: ", sb->s_id,
> +	       function, line);
>  	vprintk(fmt, args);
>  	printk("\n");
>  	va_end(args);
>  
> +	if ((sb->s_flags & MS_RDONLY) == 0) {
> +		ext4_msg(sb, KERN_CRIT, "Remounting filesystem read-only");
> +		EXT4_SB(sb)->s_mount_state |= EXT4_ERROR_FS;
> +		sb->s_flags |= MS_RDONLY;
> +		EXT4_SB(sb)->s_mount_flags |= EXT4_MF_FS_ABORTED;
> +		if (EXT4_SB(sb)->s_journal)
> +			jbd2_journal_abort(EXT4_SB(sb)->s_journal, -EIO);
> +		strncpy(es->s_error_func, function, sizeof(es->s_error_func));
> +		es->s_error_line = cpu_to_le32(line);
> +		ext4_commit_super(sb, 1);
> +	}
>  	if (test_opt(sb, ERRORS_PANIC))
>  		panic("EXT4-fs panic from previous error\n");
> -
> -	if (sb->s_flags & MS_RDONLY)
> -		return;
> -
> -	ext4_msg(sb, KERN_CRIT, "Remounting filesystem read-only");
> -	EXT4_SB(sb)->s_mount_state |= EXT4_ERROR_FS;
> -	sb->s_flags |= MS_RDONLY;
> -	EXT4_SB(sb)->s_mount_flags |= EXT4_MF_FS_ABORTED;
> -	if (EXT4_SB(sb)->s_journal)
> -		jbd2_journal_abort(EXT4_SB(sb)->s_journal, -EIO);
>  }
>  
>  void ext4_msg (struct super_block * sb, const char *prefix,
> @@ -502,20 +528,21 @@ void ext4_msg (struct super_block * sb, const char *prefix,
>  }
>  
>  void __ext4_warning(struct super_block *sb, const char *function,
> -		  const char *fmt, ...)
> +		    unsigned int line, const char *fmt, ...)
>  {
>  	va_list args;
>  
>  	va_start(args, fmt);
> -	printk(KERN_WARNING "EXT4-fs warning (device %s): %s: ",
> -	       sb->s_id, function);
> +	printk(KERN_WARNING "EXT4-fs warning (device %s): %s:%d: ",
> +	       sb->s_id, function, line);
>  	vprintk(fmt, args);
>  	printk("\n");
>  	va_end(args);
>  }
>  
> -void ext4_grp_locked_error(struct super_block *sb, ext4_group_t grp,
> -			   const char *function, const char *fmt, ...)
> +void __ext4_grp_locked_error(const char *function, unsigned int line,
> +			   struct super_block *sb, ext4_group_t grp,
> +			   const char *fmt, ...)
>  __releases(bitlock)
>  __acquires(bitlock)
>  {
> @@ -523,14 +550,21 @@ __acquires(bitlock)
>  	struct ext4_super_block *es = EXT4_SB(sb)->s_es;
>  
>  	va_start(args, fmt);
> -	printk(KERN_CRIT "EXT4-fs error (device %s): %s: ", sb->s_id, function);
> +	printk(KERN_CRIT "EXT4-fs error (device %s): %s:%d: ", sb->s_id,
> +	       function, line);
>  	vprintk(fmt, args);
>  	printk("\n");
>  	va_end(args);
> +	strncpy(es->s_error_func, function, sizeof(es->s_error_func));
> +	es->s_error_line = cpu_to_le32(line);
>  
>  	if (test_opt(sb, ERRORS_CONT)) {
>  		EXT4_SB(sb)->s_mount_state |= EXT4_ERROR_FS;
>  		es->s_state |= cpu_to_le16(EXT4_ERROR_FS);
> +		es->s_last_error_time = cpu_to_le32(get_seconds());
> +		if (!es->s_first_error_time)
> +			es->s_first_error_time = es->s_last_error_time;
> +		es->s_error_count = cpu_to_le32(le32_to_cpu(es->s_error_count) + 1);
>  		ext4_commit_super(sb, 0);
>  		return;
>  	}
> @@ -660,8 +694,7 @@ static void ext4_put_super(struct super_block *sb)
>  		err = jbd2_journal_destroy(sbi->s_journal);
>  		sbi->s_journal = NULL;
>  		if (err < 0)
> -			ext4_abort(sb, __func__,
> -				   "Couldn't clean up the journal");
> +			ext4_abort(sb, "Couldn't clean up the journal");
>  	}
>  
>  	ext4_release_system_zone(sb);
> @@ -3605,7 +3638,7 @@ static int ext4_remount(struct super_block *sb, int *flags, char *data)
>  	}
>  
>  	if (sbi->s_mount_flags & EXT4_MF_FS_ABORTED)
> -		ext4_abort(sb, __func__, "Abort forced by user");
> +		ext4_abort(sb, "Abort forced by user");
>  
>  	sb->s_flags = (sb->s_flags & ~MS_POSIXACL) |
>  		(test_opt(sb, POSIX_ACL) ? MS_POSIXACL : 0);
--
To unsubscribe from this list: send the line "unsubscribe linux-ext4" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[Index of Archives]     [Reiser Filesystem Development]     [Ceph FS]     [Kernel Newbies]     [Security]     [Netfilter]     [Bugtraq]     [Linux FS]     [Yosemite National Park]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Samba]     [Device Mapper]     [Linux Media]

  Powered by Linux