mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2025-01-25 15:24:17 +08:00
bcachefs: Add a lock to bch_page_state
We can't use the page lock to protect it, because on writeback IO error we need to access the page state before calling end_page_writeback() and the page lock semantics are completely insane so that deadlocks. Signed-off-by: Kent Overstreet <kent.overstreet@linux.dev>
This commit is contained in:
parent
43de7376f3
commit
3826ee0b17
@ -468,6 +468,7 @@ struct bch_page_sector {
|
|||||||
};
|
};
|
||||||
|
|
||||||
struct bch_page_state {
|
struct bch_page_state {
|
||||||
|
spinlock_t lock;
|
||||||
atomic_t write_count;
|
atomic_t write_count;
|
||||||
struct bch_page_sector s[PAGE_SECTORS];
|
struct bch_page_sector s[PAGE_SECTORS];
|
||||||
};
|
};
|
||||||
@ -523,6 +524,7 @@ static struct bch_page_state *__bch2_page_state_create(struct page *page,
|
|||||||
if (!s)
|
if (!s)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
spin_lock_init(&s->lock);
|
||||||
/*
|
/*
|
||||||
* migrate_page_move_mapping() assumes that pages with private data
|
* migrate_page_move_mapping() assumes that pages with private data
|
||||||
* have their count elevated by 1.
|
* have their count elevated by 1.
|
||||||
@ -670,6 +672,9 @@ static void bch2_clear_page_bits(struct page *page)
|
|||||||
if (!s)
|
if (!s)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
EBUG_ON(!PageLocked(page));
|
||||||
|
EBUG_ON(PageWriteback(page));
|
||||||
|
|
||||||
for (i = 0; i < ARRAY_SIZE(s->s); i++) {
|
for (i = 0; i < ARRAY_SIZE(s->s); i++) {
|
||||||
disk_res.sectors += s->s[i].replicas_reserved;
|
disk_res.sectors += s->s[i].replicas_reserved;
|
||||||
s->s[i].replicas_reserved = 0;
|
s->s[i].replicas_reserved = 0;
|
||||||
@ -699,6 +704,8 @@ static void bch2_set_page_dirty(struct bch_fs *c,
|
|||||||
WARN_ON((u64) page_offset(page) + offset + len >
|
WARN_ON((u64) page_offset(page) + offset + len >
|
||||||
round_up((u64) i_size_read(&inode->v), block_bytes(c)));
|
round_up((u64) i_size_read(&inode->v), block_bytes(c)));
|
||||||
|
|
||||||
|
spin_lock(&s->lock);
|
||||||
|
|
||||||
for (i = round_down(offset, block_bytes(c)) >> 9;
|
for (i = round_down(offset, block_bytes(c)) >> 9;
|
||||||
i < round_up(offset + len, block_bytes(c)) >> 9;
|
i < round_up(offset + len, block_bytes(c)) >> 9;
|
||||||
i++) {
|
i++) {
|
||||||
@ -715,6 +722,8 @@ static void bch2_set_page_dirty(struct bch_fs *c,
|
|||||||
s->s[i].state = max_t(unsigned, s->s[i].state, SECTOR_DIRTY);
|
s->s[i].state = max_t(unsigned, s->s[i].state, SECTOR_DIRTY);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
spin_unlock(&s->lock);
|
||||||
|
|
||||||
if (dirty_sectors)
|
if (dirty_sectors)
|
||||||
i_sectors_acct(c, inode, &res->quota, dirty_sectors);
|
i_sectors_acct(c, inode, &res->quota, dirty_sectors);
|
||||||
|
|
||||||
@ -798,10 +807,6 @@ void bch2_invalidate_folio(struct folio *folio, size_t offset, size_t length)
|
|||||||
|
|
||||||
bool bch2_release_folio(struct folio *folio, gfp_t gfp_mask)
|
bool bch2_release_folio(struct folio *folio, gfp_t gfp_mask)
|
||||||
{
|
{
|
||||||
/* XXX: this can't take locks that are held while we allocate memory */
|
|
||||||
EBUG_ON(!PageLocked(&folio->page));
|
|
||||||
EBUG_ON(folio_test_writeback(folio));
|
|
||||||
|
|
||||||
if (folio_test_dirty(folio) || folio_test_writeback(folio))
|
if (folio_test_dirty(folio) || folio_test_writeback(folio))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
@ -1183,11 +1188,11 @@ static void bch2_writepage_io_done(struct closure *cl)
|
|||||||
SetPageError(bvec->bv_page);
|
SetPageError(bvec->bv_page);
|
||||||
mapping_set_error(bvec->bv_page->mapping, -EIO);
|
mapping_set_error(bvec->bv_page->mapping, -EIO);
|
||||||
|
|
||||||
lock_page(bvec->bv_page);
|
s = __bch2_page_state(bvec->bv_page);
|
||||||
s = bch2_page_state(bvec->bv_page);
|
spin_lock(&s->lock);
|
||||||
for (i = 0; i < PAGE_SECTORS; i++)
|
for (i = 0; i < PAGE_SECTORS; i++)
|
||||||
s->s[i].nr_replicas = 0;
|
s->s[i].nr_replicas = 0;
|
||||||
unlock_page(bvec->bv_page);
|
spin_unlock(&s->lock);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2978,9 +2983,12 @@ static void mark_range_unallocated(struct bch_inode_info *inode,
|
|||||||
folio_lock(folio);
|
folio_lock(folio);
|
||||||
s = bch2_page_state(&folio->page);
|
s = bch2_page_state(&folio->page);
|
||||||
|
|
||||||
if (s)
|
if (s) {
|
||||||
|
spin_lock(&s->lock);
|
||||||
for (j = 0; j < PAGE_SECTORS; j++)
|
for (j = 0; j < PAGE_SECTORS; j++)
|
||||||
s->s[j].nr_replicas = 0;
|
s->s[j].nr_replicas = 0;
|
||||||
|
spin_unlock(&s->lock);
|
||||||
|
}
|
||||||
|
|
||||||
folio_unlock(folio);
|
folio_unlock(folio);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user