Re: Possible UDF locking error?
From: Jan Kara
Date: Mon Mar 25 2019 - 12:42:15 EST
Hi!
On Sat 23-03-19 15:14:05, Steve Magnani wrote:
> I have been hunting a UDF bug that occasionally results in generation
> of an Allocation Extent Descriptor with an incorrect tagLocation. So
> far I haven't been able to see a path through the code that could
> cause that. But, I noticed some inconsistency in locking during
> AED generation and wonder if it could result in random corruption.
>
> The function udf_update_inode() has this general pattern:
>
> bh = udf_tgetblk(...); // calls sb_getblk()
> lock_buffer(bh);
> memset(bh->b_data, 0, inode->i_sb->s_blocksize);
> // <snip>other code to populate FE/EFE data in the block</snip>
> set_buffer_uptodate(bh);
> unlock_buffer(bh);
> mark_buffer_dirty(bh);
>
> This I can understand - the lock is held for as long as the buffer
> contents are being assembled.
>
> In contrast, udf_setup_indirect_aext(), which constructs an AED,
> has this sequence:
>
> bh = udf_tgetblk(...); // calls sb_getblk()
> lock_buffer(bh);
> memset(bh->b_data, 0, inode->i_sb->s_blocksize);
>
> set_buffer_uptodate(bh);
> unlock_buffer(bh);
> mark_buffer_dirty_inode(bh);
>
> // <snip>other code to populate AED data in the block</snip>
>
> In this case the population of the block occurs without
> the protection of the lock.
>
> Because the block has been marked dirty, does this mean that
> writeback could occur at any point during population?
Yes. Thanks for noticing this!
> There is one path through udf_setup_indirect_aext() where
> mark_buffer_dirty_inode() gets called again after population is
> complete, which I suppose could heal a partial writeout, but there is
> also another path in which the buffer does not get marked dirty again.
Generally, we add new extents to the created indirect extent which dirties
the buffer and that should fix the problem. But you are definitely right
that the code is suspicious and should be fixed. Will you send a patch?
Honza
--
Jan Kara <jack@xxxxxxxx>
SUSE Labs, CR