Linux: osi_vcache: Fix loop for the hlist case
[openafs.git] / src / afs / LINUX / osi_vcache.c
1 /*
2  * Copyright 2000, International Business Machines Corporation and others.
3  * All Rights Reserved.
4  *
5  * This software has been released under the terms of the IBM Public
6  * License.  For details, see the LICENSE file in the top-level source
7  * directory or online at http://www.openafs.org/dl/license10.html
8  */
9
10 #include <afsconfig.h>
11 #include "afs/param.h"
12
13 #include "afs/sysincludes.h"    /*Standard vendor system headers */
14 #include "afsincludes.h"        /*AFS-based standard headers */
15
16 int
17 osi_TryEvictVCache(struct vcache *avc, int *slept, int defersleep) {
18     int code;
19
20     struct dentry *dentry;
21     struct inode *inode = AFSTOV(avc);
22 #if defined(D_ALIAS_IS_HLIST)
23     struct hlist_node *cur, *head, *list_end;
24 #else
25     struct list_head *cur, *head, *list_end;
26 #endif
27
28     /* First, see if we can evict the inode from the dcache */
29     if (defersleep && avc != afs_globalVp && VREFCOUNT(avc) > 1 && avc->opens == 0) {
30         *slept = 1;
31         ReleaseWriteLock(&afs_xvcache);
32         AFS_GUNLOCK();
33
34 #if defined(HAVE_DCACHE_LOCK)
35         spin_lock(&dcache_lock);
36         head = &inode->i_dentry;
37
38 restart:
39         cur = head;
40         while ((cur = cur->next) != head) {
41             dentry = list_entry(cur, struct dentry, d_alias);
42
43             if (d_unhashed(dentry))
44                 continue;
45             dget_locked(dentry);
46
47             spin_unlock(&dcache_lock);
48             if (d_invalidate(dentry) == -EBUSY) {
49                 dput(dentry);
50                 /* perhaps lock and try to continue? (use cur as head?) */
51                 goto inuse;
52             }
53             dput(dentry);
54             spin_lock(&dcache_lock);
55             goto restart;
56         }
57         spin_unlock(&dcache_lock);
58 #else /* HAVE_DCACHE_LOCK */
59         spin_lock(&inode->i_lock);
60 #if defined(D_ALIAS_IS_HLIST)
61         head = inode->i_dentry.first;
62         list_end = NULL;
63 #else
64         head = &inode->i_dentry;
65         list_end = head;
66 #endif
67
68 restart:
69         cur = head;
70         while ((cur = cur->next) != list_end) {
71 #if defined(D_ALIAS_IS_HLIST)
72             dentry = hlist_entry(cur, struct dentry, d_alias);
73 #else
74             dentry = list_entry(cur, struct dentry, d_alias);
75 #endif
76
77             spin_lock(&dentry->d_lock);
78             if (d_unhashed(dentry)) {
79                 spin_unlock(&dentry->d_lock);
80                 continue;
81             }
82             spin_unlock(&dentry->d_lock);
83             dget(dentry);
84
85             spin_unlock(&inode->i_lock);
86             if (d_invalidate(dentry) == -EBUSY) {
87                 dput(dentry);
88                 /* perhaps lock and try to continue? (use cur as head?) */
89                 goto inuse;
90             }
91             dput(dentry);
92             spin_lock(&inode->i_lock);
93             goto restart;
94         }
95         spin_unlock(&inode->i_lock);
96 #endif /* HAVE_DCACHE_LOCK */
97 inuse:
98         AFS_GLOCK();
99         ObtainWriteLock(&afs_xvcache, 733);
100     }
101
102     /* See if we can evict it from the VLRUQ */
103     if (VREFCOUNT_GT(avc,0) && !VREFCOUNT_GT(avc,1) && avc->opens == 0
104         && (avc->f.states & CUnlinkedDel) == 0) {
105         int didsleep = *slept;
106
107         code = afs_FlushVCache(avc, slept);
108         /* flushvcache wipes slept; restore slept if we did before */
109         if (didsleep)
110             *slept = didsleep;
111
112         if (code == 0)
113             return 1;
114     }
115
116     return 0;
117 }
118
119 struct vcache *
120 osi_NewVnode(void)
121 {
122     struct inode *ip;
123     struct vcache *tvc;
124
125     AFS_GUNLOCK();
126     ip = new_inode(afs_globalVFS);
127     if (!ip)
128         osi_Panic("afs_NewVCache: no more inodes");
129     AFS_GLOCK();
130 #if defined(STRUCT_SUPER_OPERATIONS_HAS_ALLOC_INODE)
131     tvc = VTOAFS(ip);
132 #else
133     tvc = afs_osi_Alloc(sizeof(struct vcache));
134     ip->u.generic_ip = tvc;
135     tvc->v = ip;
136 #endif
137
138     return tvc;
139 }
140
141 void
142 osi_PrePopulateVCache(struct vcache *avc) {
143     avc->uncred = 0;
144     memset(&(avc->f), 0, sizeof(struct fvcache));
145     avc->cred = NULL;
146 }
147
148 void
149 osi_AttachVnode(struct vcache *avc, int seq) { /* Nada */ }
150
151 void
152 osi_PostPopulateVCache(struct vcache *avc) {
153     vSetType(avc, VREG);
154 }
155