mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-17 10:46:33 +00:00
io-wq: serialize hash clear with wakeup
We need to ensure that we serialize the stalled and hash bits with the wait_queue wait handler, or we could be racing with someone modifying the hashed state after we find it busy, but before we then give up and wait for it to be cleared. This can cause random delays or stalls when handling buffered writes for many files, where some of these files cause hash collisions between the worker threads. Cc: stable@vger.kernel.org Reported-by: Daniel Black <daniel@mariadb.org> Fixes: e941894eae31 ("io-wq: make buffered file write hashed work map per-ctx") Signed-off-by: Jens Axboe <axboe@kernel.dk>
This commit is contained in:
parent
bad119b9a0
commit
d3e3c102d1
17
fs/io-wq.c
17
fs/io-wq.c
@ -423,9 +423,10 @@ static inline unsigned int io_get_work_hash(struct io_wq_work *work)
|
||||
return work->flags >> IO_WQ_HASH_SHIFT;
|
||||
}
|
||||
|
||||
static void io_wait_on_hash(struct io_wqe *wqe, unsigned int hash)
|
||||
static bool io_wait_on_hash(struct io_wqe *wqe, unsigned int hash)
|
||||
{
|
||||
struct io_wq *wq = wqe->wq;
|
||||
bool ret = false;
|
||||
|
||||
spin_lock_irq(&wq->hash->wait.lock);
|
||||
if (list_empty(&wqe->wait.entry)) {
|
||||
@ -433,9 +434,11 @@ static void io_wait_on_hash(struct io_wqe *wqe, unsigned int hash)
|
||||
if (!test_bit(hash, &wq->hash->map)) {
|
||||
__set_current_state(TASK_RUNNING);
|
||||
list_del_init(&wqe->wait.entry);
|
||||
ret = true;
|
||||
}
|
||||
}
|
||||
spin_unlock_irq(&wq->hash->wait.lock);
|
||||
return ret;
|
||||
}
|
||||
|
||||
static struct io_wq_work *io_get_next_work(struct io_wqe_acct *acct,
|
||||
@ -475,14 +478,21 @@ static struct io_wq_work *io_get_next_work(struct io_wqe_acct *acct,
|
||||
}
|
||||
|
||||
if (stall_hash != -1U) {
|
||||
bool unstalled;
|
||||
|
||||
/*
|
||||
* Set this before dropping the lock to avoid racing with new
|
||||
* work being added and clearing the stalled bit.
|
||||
*/
|
||||
set_bit(IO_ACCT_STALLED_BIT, &acct->flags);
|
||||
raw_spin_unlock(&wqe->lock);
|
||||
io_wait_on_hash(wqe, stall_hash);
|
||||
unstalled = io_wait_on_hash(wqe, stall_hash);
|
||||
raw_spin_lock(&wqe->lock);
|
||||
if (unstalled) {
|
||||
clear_bit(IO_ACCT_STALLED_BIT, &acct->flags);
|
||||
if (wq_has_sleeper(&wqe->wq->hash->wait))
|
||||
wake_up(&wqe->wq->hash->wait);
|
||||
}
|
||||
}
|
||||
|
||||
return NULL;
|
||||
@ -564,8 +574,11 @@ get_next:
|
||||
io_wqe_enqueue(wqe, linked);
|
||||
|
||||
if (hash != -1U && !next_hashed) {
|
||||
/* serialize hash clear with wake_up() */
|
||||
spin_lock_irq(&wq->hash->wait.lock);
|
||||
clear_bit(hash, &wq->hash->map);
|
||||
clear_bit(IO_ACCT_STALLED_BIT, &acct->flags);
|
||||
spin_unlock_irq(&wq->hash->wait.lock);
|
||||
if (wq_has_sleeper(&wq->hash->wait))
|
||||
wake_up(&wq->hash->wait);
|
||||
raw_spin_lock(&wqe->lock);
|
||||
|
Loading…
x
Reference in New Issue
Block a user