707bcd3313f4017367f4764a8067dd16a8d95f84
[openafs.git] / src / afs / FBSD / 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 {
19     struct vnode *vp;
20     int code;
21
22     vp = AFSTOV(avc);
23
24     if (!VI_TRYLOCK(vp))
25         return 0;
26     code = osi_fbsd_checkinuse(avc);
27     if (code != 0) {
28         VI_UNLOCK(vp);
29         return 0;
30     }
31
32     if ((vp->v_iflag & VI_DOOMED) != 0) {
33         VI_UNLOCK(vp);
34         return 1;
35     }
36
37     /* must hold the vnode before calling vgone()
38      * This code largely copied from vfs_subr.c:vlrureclaim() */
39     vholdl(vp);
40
41     ReleaseWriteLock(&afs_xvcache);
42     AFS_GUNLOCK();
43
44     *slept = 1;
45     /* use the interlock while locking, so no one else can DOOM this */
46     vn_lock(vp, LK_INTERLOCK|LK_EXCLUSIVE|LK_RETRY);
47     vgone(vp);
48     VOP_UNLOCK(vp, 0);
49     vdrop(vp);
50
51     AFS_GLOCK();
52     ObtainWriteLock(&afs_xvcache, 340);
53     return 1;
54 }
55
56 struct vcache *
57 osi_NewVnode(void)
58 {
59     struct vcache *tvc;
60
61     tvc = afs_osi_Alloc(sizeof(struct vcache));
62     tvc->v = NULL; /* important to clean this, or use memset 0 */
63
64     return tvc;
65 }
66
67 void
68 osi_PrePopulateVCache(struct vcache *avc)
69 {
70     memset(avc, 0, sizeof(struct vcache));
71 }
72
73 void
74 osi_AttachVnode(struct vcache *avc, int seq)
75 {
76     struct vnode *vp;
77
78     ReleaseWriteLock(&afs_xvcache);
79     AFS_GUNLOCK();
80     if (getnewvnode(MOUNT_AFS, afs_globalVFS, &afs_vnodeops, &vp))
81         panic("afs getnewvnode");       /* can't happen */
82     /* XXX verified on 80--TODO check on 7x */
83     if (!vp->v_mount) {
84         vn_lock(vp, LK_EXCLUSIVE | LK_RETRY); /* !glocked */
85         insmntque(vp, afs_globalVFS);
86         VOP_UNLOCK(vp, 0);
87     }
88     AFS_GLOCK();
89     ObtainWriteLock(&afs_xvcache,339);
90     if (avc->v != NULL) {
91         /* I'd like to know if this ever happens...
92          * We don't drop global for the rest of this function,
93          * so if we do lose the race, the other thread should
94          * have found the same vnode and finished initializing
95          * the vcache entry.  Is it conceivable that this vcache
96          * entry could be recycled during this interval?  If so,
97          * then there probably needs to be some sort of additional
98          * mutual exclusion (an Embryonic flag would suffice).
99          * -GAW */
100         afs_warn("afs_NewVCache: lost the race\n");
101         return;
102     }
103     avc->v = vp;
104     avc->v->v_data = avc;
105     lockinit(&avc->rwlock, PINOD, "vcache", 0, 0);
106 }
107
108 void
109 osi_PostPopulateVCache(struct vcache *avc)
110 {
111     avc->v->v_mount = afs_globalVFS;
112     vSetType(avc, VREG);
113 }
114
115 int
116 osi_vnhold(struct vcache *avc)
117 {
118     vref(AFSTOV(avc));
119     return 0;
120 }