Commit 19fe87e0 authored by Kent Overstreet's avatar Kent Overstreet

bcachefs: Inline bch2_two_state_(trylock|unlock)

Standard inlining of fast paths - these locks are now used by our new
nocow mode.
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent a8b3a677
...@@ -10,6 +10,6 @@ void __bch2_bucket_nocow_lock(struct bucket_nocow_lock_table *t, ...@@ -10,6 +10,6 @@ void __bch2_bucket_nocow_lock(struct bucket_nocow_lock_table *t,
struct bch_fs *c = container_of(t, struct bch_fs, nocow_locks); struct bch_fs *c = container_of(t, struct bch_fs, nocow_locks);
u64 start_time = local_clock(); u64 start_time = local_clock();
bch2_two_state_lock(l, flags & BUCKET_NOCOW_LOCK_UPDATE); __bch2_two_state_lock(l, flags & BUCKET_NOCOW_LOCK_UPDATE);
bch2_time_stats_update(&c->times[BCH_TIME_nocow_lock_contended], start_time); bch2_time_stats_update(&c->times[BCH_TIME_nocow_lock_contended], start_time);
} }
...@@ -2,32 +2,7 @@ ...@@ -2,32 +2,7 @@
#include "two_state_shared_lock.h" #include "two_state_shared_lock.h"
void bch2_two_state_unlock(two_state_lock_t *lock, int s) void __bch2_two_state_lock(two_state_lock_t *lock, int s)
{ {
long i = s ? 1 : -1; __wait_event(lock->wait, bch2_two_state_trylock(lock, s));
BUG_ON(atomic_long_read(&lock->v) == 0);
if (atomic_long_sub_return_release(i, &lock->v) == 0)
wake_up_all(&lock->wait);
}
bool bch2_two_state_trylock(two_state_lock_t *lock, int s)
{
long i = s ? 1 : -1;
long v = atomic_long_read(&lock->v), old;
do {
old = v;
if (i > 0 ? v < 0 : v > 0)
return false;
} while ((v = atomic_long_cmpxchg_acquire(&lock->v,
old, old + i)) != old);
return true;
}
void bch2_two_state_lock(two_state_lock_t *lock, int s)
{
wait_event(lock->wait, bch2_two_state_trylock(lock, s));
} }
...@@ -6,6 +6,8 @@ ...@@ -6,6 +6,8 @@
#include <linux/sched.h> #include <linux/sched.h>
#include <linux/wait.h> #include <linux/wait.h>
#include "util.h"
/* /*
* Two-state lock - can be taken for add or block - both states are shared, * Two-state lock - can be taken for add or block - both states are shared,
* like read side of rwsem, but conflict with other state: * like read side of rwsem, but conflict with other state:
...@@ -21,8 +23,37 @@ static inline void two_state_lock_init(two_state_lock_t *lock) ...@@ -21,8 +23,37 @@ static inline void two_state_lock_init(two_state_lock_t *lock)
init_waitqueue_head(&lock->wait); init_waitqueue_head(&lock->wait);
} }
void bch2_two_state_unlock(two_state_lock_t *, int); static inline void bch2_two_state_unlock(two_state_lock_t *lock, int s)
bool bch2_two_state_trylock(two_state_lock_t *, int); {
void bch2_two_state_lock(two_state_lock_t *, int); long i = s ? 1 : -1;
EBUG_ON(atomic_long_read(&lock->v) == 0);
if (atomic_long_sub_return_release(i, &lock->v) == 0)
wake_up_all(&lock->wait);
}
static inline bool bch2_two_state_trylock(two_state_lock_t *lock, int s)
{
long i = s ? 1 : -1;
long v = atomic_long_read(&lock->v), old;
do {
old = v;
if (i > 0 ? v < 0 : v > 0)
return false;
} while ((v = atomic_long_cmpxchg_acquire(&lock->v,
old, old + i)) != old);
return true;
}
void __bch2_two_state_lock(two_state_lock_t *, int);
static inline void bch2_two_state_lock(two_state_lock_t *lock, int s)
{
if (!bch2_two_state_trylock(lock, s))
__bch2_two_state_lock(lock, s);
}
#endif /* _BCACHEFS_TWO_STATE_LOCK_H */ #endif /* _BCACHEFS_TWO_STATE_LOCK_H */
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