linux-fsdevel.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [rfc][patch] fs: block_dump missing dentry locking
@ 2009-05-28  7:01 Nick Piggin
  2009-05-28 12:25 ` Josef Bacik
  0 siblings, 1 reply; 2+ messages in thread
From: Nick Piggin @ 2009-05-28  7:01 UTC (permalink / raw)
  To: Al Viro, linux-fsdevel; +Cc: Jens Axboe

I think the block_dump output in __mark_inode_dirty is missing dentry locking.
Surely the i_dentry list can change any time, so we may not even *get* a
dentry there. If we do get one by chance, then it would appear to be able to
go away or get renamed at any time...

I've just gone the very-safe route and used the dcache helper to find this,
and done the full refcounting/locking thing. Also just moved it to its
own function to reduce clutter a bit.

Is my thinking correct?

Thanks,
Nick
---
 fs/fs-writeback.c |   41 ++++++++++++++++++++++++-----------------
 1 file changed, 24 insertions(+), 17 deletions(-)

Index: linux-2.6/fs/fs-writeback.c
===================================================================
--- linux-2.6.orig/fs/fs-writeback.c
+++ linux-2.6/fs/fs-writeback.c
@@ -64,6 +64,28 @@ static void writeback_release(struct bac
 	clear_bit(BDI_pdflush, &bdi->state);
 }
 
+static noinline void block_dump___mark_inode_dirty(struct inode *inode)
+{
+	if (inode->i_ino || strcmp(inode->i_sb->s_id, "bdev")) {
+		struct dentry *dentry;
+		const char *name = "?";
+
+		dentry = d_find_alias(inode);
+		if (dentry) {
+			spin_lock(&dentry->d_lock);
+			name = (const char *) dentry->d_name.name;
+		}
+		printk(KERN_DEBUG
+		       "%s(%d): dirtied inode %lu (%s) on %s\n",
+		       current->comm, task_pid_nr(current), inode->i_ino,
+		       name, inode->i_sb->s_id);
+		if (dentry) {
+			spin_unlock(&dentry->d_lock);
+			dput(dentry);
+		}
+	}
+}
+
 /**
  *	__mark_inode_dirty -	internal function
  *	@inode: inode to mark
@@ -114,23 +136,8 @@ void __mark_inode_dirty(struct inode *in
 	if ((inode->i_state & flags) == flags)
 		return;
 
-	if (unlikely(block_dump)) {
-		struct dentry *dentry = NULL;
-		const char *name = "?";
-
-		if (!list_empty(&inode->i_dentry)) {
-			dentry = list_entry(inode->i_dentry.next,
-					    struct dentry, d_alias);
-			if (dentry && dentry->d_name.name)
-				name = (const char *) dentry->d_name.name;
-		}
-
-		if (inode->i_ino || strcmp(inode->i_sb->s_id, "bdev"))
-			printk(KERN_DEBUG
-			       "%s(%d): dirtied inode %lu (%s) on %s\n",
-			       current->comm, task_pid_nr(current), inode->i_ino,
-			       name, inode->i_sb->s_id);
-	}
+	if (unlikely(block_dump))
+		block_dump___mark_inode_dirty(inode);
 
 	spin_lock(&inode_lock);
 	if ((inode->i_state & flags) != flags) {

^ permalink raw reply	[flat|nested] 2+ messages in thread

* Re: [rfc][patch] fs: block_dump missing dentry locking
  2009-05-28  7:01 [rfc][patch] fs: block_dump missing dentry locking Nick Piggin
@ 2009-05-28 12:25 ` Josef Bacik
  0 siblings, 0 replies; 2+ messages in thread
From: Josef Bacik @ 2009-05-28 12:25 UTC (permalink / raw)
  To: Nick Piggin; +Cc: Al Viro, linux-fsdevel, Jens Axboe

On Thu, May 28, 2009 at 09:01:15AM +0200, Nick Piggin wrote:
> I think the block_dump output in __mark_inode_dirty is missing dentry locking.
> Surely the i_dentry list can change any time, so we may not even *get* a
> dentry there. If we do get one by chance, then it would appear to be able to
> go away or get renamed at any time...
> 
> I've just gone the very-safe route and used the dcache helper to find this,
> and done the full refcounting/locking thing. Also just moved it to its
> own function to reduce clutter a bit.
> 
> Is my thinking correct?
> 
> Thanks,
> Nick
> ---
>  fs/fs-writeback.c |   41 ++++++++++++++++++++++++-----------------
>  1 file changed, 24 insertions(+), 17 deletions(-)
> 
> Index: linux-2.6/fs/fs-writeback.c
> ===================================================================
> --- linux-2.6.orig/fs/fs-writeback.c
> +++ linux-2.6/fs/fs-writeback.c
> @@ -64,6 +64,28 @@ static void writeback_release(struct bac
>  	clear_bit(BDI_pdflush, &bdi->state);
>  }
>  
> +static noinline void block_dump___mark_inode_dirty(struct inode *inode)
> +{
> +	if (inode->i_ino || strcmp(inode->i_sb->s_id, "bdev")) {
> +		struct dentry *dentry;
> +		const char *name = "?";
> +
> +		dentry = d_find_alias(inode);
> +		if (dentry) {
> +			spin_lock(&dentry->d_lock);
> +			name = (const char *) dentry->d_name.name;
> +		}
> +		printk(KERN_DEBUG
> +		       "%s(%d): dirtied inode %lu (%s) on %s\n",
> +		       current->comm, task_pid_nr(current), inode->i_ino,
> +		       name, inode->i_sb->s_id);
> +		if (dentry) {
> +			spin_unlock(&dentry->d_lock);
> +			dput(dentry);
> +		}
> +	}
> +}
> +
>  /**
>   *	__mark_inode_dirty -	internal function
>   *	@inode: inode to mark
> @@ -114,23 +136,8 @@ void __mark_inode_dirty(struct inode *in
>  	if ((inode->i_state & flags) == flags)
>  		return;
>  
> -	if (unlikely(block_dump)) {
> -		struct dentry *dentry = NULL;
> -		const char *name = "?";
> -
> -		if (!list_empty(&inode->i_dentry)) {
> -			dentry = list_entry(inode->i_dentry.next,
> -					    struct dentry, d_alias);
> -			if (dentry && dentry->d_name.name)
> -				name = (const char *) dentry->d_name.name;
> -		}
> -
> -		if (inode->i_ino || strcmp(inode->i_sb->s_id, "bdev"))
> -			printk(KERN_DEBUG
> -			       "%s(%d): dirtied inode %lu (%s) on %s\n",
> -			       current->comm, task_pid_nr(current), inode->i_ino,
> -			       name, inode->i_sb->s_id);
> -	}
> +	if (unlikely(block_dump))
> +		block_dump___mark_inode_dirty(inode);
>  
>  	spin_lock(&inode_lock);
>  	if ((inode->i_state & flags) != flags) {

It looks correct to me, you need to hold the dcache_lock to access the i_dentry
list safely, which you accomplish with d_find_alias, and you need to hold the
d_lock in order to safely access d_name, so it all looks right.  Thanks,

Josef

^ permalink raw reply	[flat|nested] 2+ messages in thread

end of thread, other threads:[~2009-05-28 12:25 UTC | newest]

Thread overview: 2+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2009-05-28  7:01 [rfc][patch] fs: block_dump missing dentry locking Nick Piggin
2009-05-28 12:25 ` Josef Bacik

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).