Commit 671cc8a5 authored by Kent Overstreet's avatar Kent Overstreet Committed by Kent Overstreet

bcachefs: Eliminate memory barrier from fast path of journal_preres_put()

Signed-off-by: default avatarKent Overstreet <kent.overstreet@gmail.com>
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent 08e33761
...@@ -520,7 +520,7 @@ static bool journal_preres_available(struct journal *j, ...@@ -520,7 +520,7 @@ static bool journal_preres_available(struct journal *j,
unsigned new_u64s, unsigned new_u64s,
unsigned flags) unsigned flags)
{ {
bool ret = bch2_journal_preres_get_fast(j, res, new_u64s, flags); bool ret = bch2_journal_preres_get_fast(j, res, new_u64s, flags, true);
if (!ret && mutex_trylock(&j->reclaim_lock)) { if (!ret && mutex_trylock(&j->reclaim_lock)) {
bch2_journal_reclaim(j); bch2_journal_reclaim(j);
......
...@@ -411,7 +411,12 @@ static inline void bch2_journal_preres_put(struct journal *j, ...@@ -411,7 +411,12 @@ static inline void bch2_journal_preres_put(struct journal *j,
s.v = atomic64_sub_return(s.v, &j->prereserved.counter); s.v = atomic64_sub_return(s.v, &j->prereserved.counter);
res->u64s = 0; res->u64s = 0;
closure_wake_up(&j->preres_wait);
if (unlikely(s.waiting)) {
clear_bit(ilog2((((union journal_preres_state) { .waiting = 1 }).v)),
(unsigned long *) &j->prereserved.v);
closure_wake_up(&j->preres_wait);
}
if (s.reserved <= s.remaining && if (s.reserved <= s.remaining &&
!test_bit(JOURNAL_MAY_GET_UNRESERVED, &j->flags)) { !test_bit(JOURNAL_MAY_GET_UNRESERVED, &j->flags)) {
...@@ -427,32 +432,32 @@ int __bch2_journal_preres_get(struct journal *, ...@@ -427,32 +432,32 @@ int __bch2_journal_preres_get(struct journal *,
static inline int bch2_journal_preres_get_fast(struct journal *j, static inline int bch2_journal_preres_get_fast(struct journal *j,
struct journal_preres *res, struct journal_preres *res,
unsigned new_u64s, unsigned new_u64s,
unsigned flags) unsigned flags,
bool set_waiting)
{ {
int d = new_u64s - res->u64s; int d = new_u64s - res->u64s;
union journal_preres_state old, new; union journal_preres_state old, new;
u64 v = atomic64_read(&j->prereserved.counter); u64 v = atomic64_read(&j->prereserved.counter);
int ret;
do { do {
old.v = new.v = v; old.v = new.v = v;
ret = 0;
new.reserved += d;
if ((flags & JOURNAL_RES_GET_RESERVED) ||
/* new.reserved + d < new.remaining) {
* If we're being called from the journal reclaim path, we have new.reserved += d;
* to unconditionally give out the pre-reservation, there's ret = 1;
* nothing else sensible we can do - otherwise we'd recurse back } else if (set_waiting && !new.waiting)
* into the reclaim path and deadlock: new.waiting = true;
*/ else
if (!(flags & JOURNAL_RES_GET_RESERVED) &&
new.reserved > new.remaining)
return 0; return 0;
} while ((v = atomic64_cmpxchg(&j->prereserved.counter, } while ((v = atomic64_cmpxchg(&j->prereserved.counter,
old.v, new.v)) != old.v); old.v, new.v)) != old.v);
res->u64s += d; if (ret)
return 1; res->u64s += d;
return ret;
} }
static inline int bch2_journal_preres_get(struct journal *j, static inline int bch2_journal_preres_get(struct journal *j,
...@@ -463,7 +468,7 @@ static inline int bch2_journal_preres_get(struct journal *j, ...@@ -463,7 +468,7 @@ static inline int bch2_journal_preres_get(struct journal *j,
if (new_u64s <= res->u64s) if (new_u64s <= res->u64s)
return 0; return 0;
if (bch2_journal_preres_get_fast(j, res, new_u64s, flags)) if (bch2_journal_preres_get_fast(j, res, new_u64s, flags, false))
return 0; return 0;
if (flags & JOURNAL_RES_GET_NONBLOCK) if (flags & JOURNAL_RES_GET_NONBLOCK)
......
...@@ -105,8 +105,9 @@ union journal_preres_state { ...@@ -105,8 +105,9 @@ union journal_preres_state {
}; };
struct { struct {
u32 reserved; u64 waiting:1,
u32 remaining; reserved:31,
remaining:32;
}; };
}; };
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment