Commit acf1f42f authored by Andreas Gruenbacher's avatar Andreas Gruenbacher

gfs2: Fix "Make glock lru list scanning safer"

Commit 228804a3 tried to add a refcount check to
gfs2_scan_glock_lru() to make sure that glocks that are still referenced
cannot be freed.  It failed to account for the bias state_change() adds
to the refcount for held glocks, so held glocks are no longer removed
from the glock cache, which can lead to out-of-memory problems.  Fix
that.  (The inodes those glocks are associated with do get shrunk and do
get pushed out of memory.)

In addition, use the same eligibility check in gfs2_scan_glock_lru() and
gfs2_dispose_glock_lru().
Signed-off-by: default avatarAndreas Gruenbacher <agruenba@redhat.com>
parent c9a0a4b0
...@@ -1987,6 +1987,14 @@ static int glock_cmp(void *priv, const struct list_head *a, ...@@ -1987,6 +1987,14 @@ static int glock_cmp(void *priv, const struct list_head *a,
return 0; return 0;
} }
static bool can_free_glock(struct gfs2_glock *gl)
{
bool held = gl->gl_state != LM_ST_UNLOCKED;
return !test_bit(GLF_LOCK, &gl->gl_flags) &&
gl->gl_lockref.count == held;
}
/** /**
* gfs2_dispose_glock_lru - Demote a list of glocks * gfs2_dispose_glock_lru - Demote a list of glocks
* @list: The list to dispose of * @list: The list to dispose of
...@@ -2020,7 +2028,7 @@ __acquires(&lru_lock) ...@@ -2020,7 +2028,7 @@ __acquires(&lru_lock)
atomic_inc(&lru_count); atomic_inc(&lru_count);
continue; continue;
} }
if (test_bit(GLF_LOCK, &gl->gl_flags)) { if (!can_free_glock(gl)) {
spin_unlock(&gl->gl_lockref.lock); spin_unlock(&gl->gl_lockref.lock);
goto add_back_to_lru; goto add_back_to_lru;
} }
...@@ -2052,17 +2060,11 @@ static long gfs2_scan_glock_lru(int nr) ...@@ -2052,17 +2060,11 @@ static long gfs2_scan_glock_lru(int nr)
list_for_each_entry_safe(gl, next, &lru_list, gl_lru) { list_for_each_entry_safe(gl, next, &lru_list, gl_lru) {
if (nr-- <= 0) if (nr-- <= 0)
break; break;
/* Test for being demotable */ if (can_free_glock(gl)) {
if (!test_bit(GLF_LOCK, &gl->gl_flags)) {
if (!spin_trylock(&gl->gl_lockref.lock))
continue;
if (!gl->gl_lockref.count) {
list_move(&gl->gl_lru, &dispose); list_move(&gl->gl_lru, &dispose);
atomic_dec(&lru_count); atomic_dec(&lru_count);
freed++; freed++;
} }
spin_unlock(&gl->gl_lockref.lock);
}
} }
if (!list_empty(&dispose)) if (!list_empty(&dispose))
gfs2_dispose_glock_lru(&dispose); gfs2_dispose_glock_lru(&dispose);
......
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