On Tue, Sep 15, 2020 at 03:46:16PM +0100, Matthew Wilcox (Oracle) wrote: > We may get tail pages returned from vfs_dedupe_get_page(). If we do, > we have to call page_mapping() instead of dereferencing page->mapping > directly. We may also deadlock trying to lock the page twice if they're > subpages of the same THP, so compare the head pages instead. > > Signed-off-by: Matthew Wilcox (Oracle) <willy@xxxxxxxxxxxxx> Seems fine to me... Reviewed-by: Darrick J. Wong <darrick.wong@xxxxxxxxxx> --D > --- > fs/read_write.c | 8 ++++++-- > 1 file changed, 6 insertions(+), 2 deletions(-) > > diff --git a/fs/read_write.c b/fs/read_write.c > index 5db58b8c78d0..c4d5eb47a21e 100644 > --- a/fs/read_write.c > +++ b/fs/read_write.c > @@ -1906,6 +1906,8 @@ static struct page *vfs_dedupe_get_page(struct inode *inode, loff_t offset) > */ > static void vfs_lock_two_pages(struct page *page1, struct page *page2) > { > + page1 = thp_head(page1); > + page2 = thp_head(page2); > /* Always lock in order of increasing index. */ > if (page1->index > page2->index) > swap(page1, page2); > @@ -1918,6 +1920,8 @@ static void vfs_lock_two_pages(struct page *page1, struct page *page2) > /* Unlock two pages, being careful not to unlock the same page twice. */ > static void vfs_unlock_two_pages(struct page *page1, struct page *page2) > { > + page1 = thp_head(page1); > + page2 = thp_head(page2); > unlock_page(page1); > if (page1 != page2) > unlock_page(page2); > @@ -1972,8 +1976,8 @@ static int vfs_dedupe_file_range_compare(struct inode *src, loff_t srcoff, > * someone is invalidating pages on us and we lose. > */ > if (!PageUptodate(src_page) || !PageUptodate(dest_page) || > - src_page->mapping != src->i_mapping || > - dest_page->mapping != dest->i_mapping) { > + page_mapping(src_page) != src->i_mapping || > + page_mapping(dest_page) != dest->i_mapping) { > same = false; > goto unlock; > } > -- > 2.28.0 >