aboutsummaryrefslogtreecommitdiffstats
path: root/fs/gfs2/glock.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/gfs2/glock.c')
-rw-r--r--fs/gfs2/glock.c28
1 files changed, 16 insertions, 12 deletions
diff --git a/fs/gfs2/glock.c b/fs/gfs2/glock.c
index b5effb9e4a38..cf54b0b001fd 100644
--- a/fs/gfs2/glock.c
+++ b/fs/gfs2/glock.c
@@ -40,7 +40,7 @@ struct greedy {
40}; 40};
41 41
42struct gfs2_gl_hash_bucket { 42struct gfs2_gl_hash_bucket {
43 struct list_head hb_list; 43 struct hlist_head hb_list;
44}; 44};
45 45
46typedef void (*glock_examiner) (struct gfs2_glock * gl); 46typedef void (*glock_examiner) (struct gfs2_glock * gl);
@@ -49,7 +49,7 @@ static int gfs2_dump_lockstate(struct gfs2_sbd *sdp);
49static int dump_glock(struct gfs2_glock *gl); 49static int dump_glock(struct gfs2_glock *gl);
50static int dump_inode(struct gfs2_inode *ip); 50static int dump_inode(struct gfs2_inode *ip);
51 51
52#define GFS2_GL_HASH_SHIFT 13 52#define GFS2_GL_HASH_SHIFT 15
53#define GFS2_GL_HASH_SIZE (1 << GFS2_GL_HASH_SHIFT) 53#define GFS2_GL_HASH_SIZE (1 << GFS2_GL_HASH_SHIFT)
54#define GFS2_GL_HASH_MASK (GFS2_GL_HASH_SIZE - 1) 54#define GFS2_GL_HASH_MASK (GFS2_GL_HASH_SIZE - 1)
55 55
@@ -210,7 +210,7 @@ int gfs2_glock_put(struct gfs2_glock *gl)
210 210
211 write_lock(gl_lock_addr(gl->gl_hash)); 211 write_lock(gl_lock_addr(gl->gl_hash));
212 if (kref_put(&gl->gl_ref, kill_glock)) { 212 if (kref_put(&gl->gl_ref, kill_glock)) {
213 list_del_init(&gl->gl_list); 213 hlist_del(&gl->gl_list);
214 write_unlock(gl_lock_addr(gl->gl_hash)); 214 write_unlock(gl_lock_addr(gl->gl_hash));
215 BUG_ON(spin_is_locked(&gl->gl_spin)); 215 BUG_ON(spin_is_locked(&gl->gl_spin));
216 glock_free(gl); 216 glock_free(gl);
@@ -259,8 +259,9 @@ static struct gfs2_glock *search_bucket(unsigned int hash,
259 const struct lm_lockname *name) 259 const struct lm_lockname *name)
260{ 260{
261 struct gfs2_glock *gl; 261 struct gfs2_glock *gl;
262 struct hlist_node *h;
262 263
263 list_for_each_entry(gl, &gl_hash_table[hash].hb_list, gl_list) { 264 hlist_for_each_entry(gl, h, &gl_hash_table[hash].hb_list, gl_list) {
264 if (!lm_name_equal(&gl->gl_name, name)) 265 if (!lm_name_equal(&gl->gl_name, name))
265 continue; 266 continue;
266 if (gl->gl_sbd != sdp) 267 if (gl->gl_sbd != sdp)
@@ -368,7 +369,7 @@ int gfs2_glock_get(struct gfs2_sbd *sdp, u64 number,
368 glock_free(gl); 369 glock_free(gl);
369 gl = tmp; 370 gl = tmp;
370 } else { 371 } else {
371 list_add_tail(&gl->gl_list, &gl_hash_table[hash].hb_list); 372 hlist_add_head(&gl->gl_list, &gl_hash_table[hash].hb_list);
372 write_unlock(gl_lock_addr(hash)); 373 write_unlock(gl_lock_addr(hash));
373 } 374 }
374 375
@@ -1895,15 +1896,15 @@ static int examine_bucket(glock_examiner examiner, struct gfs2_sbd *sdp,
1895{ 1896{
1896 struct gfs2_glock *gl, *prev = NULL; 1897 struct gfs2_glock *gl, *prev = NULL;
1897 int has_entries = 0; 1898 int has_entries = 0;
1898 struct list_head *head = &gl_hash_table[hash].hb_list; 1899 struct hlist_head *head = &gl_hash_table[hash].hb_list;
1899 1900
1900 read_lock(gl_lock_addr(hash)); 1901 read_lock(gl_lock_addr(hash));
1901 /* Can't use list_for_each_entry - don't want prefetch here */ 1902 /* Can't use hlist_for_each_entry - don't want prefetch here */
1902 if (list_empty(head)) 1903 if (hlist_empty(head))
1903 goto out; 1904 goto out;
1904 has_entries = 1; 1905 has_entries = 1;
1905 gl = list_entry(head->next, struct gfs2_glock, gl_list); 1906 gl = list_entry(head->first, struct gfs2_glock, gl_list);
1906 while(&gl->gl_list != head) { 1907 while(1) {
1907 if (gl->gl_sbd == sdp) { 1908 if (gl->gl_sbd == sdp) {
1908 gfs2_glock_hold(gl); 1909 gfs2_glock_hold(gl);
1909 read_unlock(gl_lock_addr(hash)); 1910 read_unlock(gl_lock_addr(hash));
@@ -1913,6 +1914,8 @@ static int examine_bucket(glock_examiner examiner, struct gfs2_sbd *sdp,
1913 examiner(gl); 1914 examiner(gl);
1914 read_lock(gl_lock_addr(hash)); 1915 read_lock(gl_lock_addr(hash));
1915 } 1916 }
1917 if (gl->gl_list.next == NULL)
1918 break;
1916 gl = list_entry(gl->gl_list.next, struct gfs2_glock, gl_list); 1919 gl = list_entry(gl->gl_list.next, struct gfs2_glock, gl_list);
1917 } 1920 }
1918out: 1921out:
@@ -2195,6 +2198,7 @@ out:
2195static int gfs2_dump_lockstate(struct gfs2_sbd *sdp) 2198static int gfs2_dump_lockstate(struct gfs2_sbd *sdp)
2196{ 2199{
2197 struct gfs2_glock *gl; 2200 struct gfs2_glock *gl;
2201 struct hlist_node *h;
2198 unsigned int x; 2202 unsigned int x;
2199 int error = 0; 2203 int error = 0;
2200 2204
@@ -2202,7 +2206,7 @@ static int gfs2_dump_lockstate(struct gfs2_sbd *sdp)
2202 2206
2203 read_lock(gl_lock_addr(x)); 2207 read_lock(gl_lock_addr(x));
2204 2208
2205 list_for_each_entry(gl, &gl_hash_table[x].hb_list, gl_list) { 2209 hlist_for_each_entry(gl, h, &gl_hash_table[x].hb_list, gl_list) {
2206 if (gl->gl_sbd != sdp) 2210 if (gl->gl_sbd != sdp)
2207 continue; 2211 continue;
2208 2212
@@ -2225,7 +2229,7 @@ int __init gfs2_glock_init(void)
2225{ 2229{
2226 unsigned i; 2230 unsigned i;
2227 for(i = 0; i < GFS2_GL_HASH_SIZE; i++) { 2231 for(i = 0; i < GFS2_GL_HASH_SIZE; i++) {
2228 INIT_LIST_HEAD(&gl_hash_table[i].hb_list); 2232 INIT_HLIST_HEAD(&gl_hash_table[i].hb_list);
2229 } 2233 }
2230#ifdef GL_HASH_LOCK_SZ 2234#ifdef GL_HASH_LOCK_SZ
2231 for(i = 0; i < GL_HASH_LOCK_SZ; i++) { 2235 for(i = 0; i < GL_HASH_LOCK_SZ; i++) {