Refactor afs_NewVCache
[openafs.git] / src / afs / DARWIN / 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 struct vcache *
17 osi_NewVnode(void) {
18     struct vcache *tvc;
19
20     tvc = (struct vcache *)afs_osi_Alloc(sizeof(struct vcache));
21     tvc->v = NULL; /* important to clean this, or use memset 0 */
22
23     return tvc;
24 }
25
26
27 #if defined(AFS_DARWIN80_ENV)
28 int
29 osi_TryEvictVCache(struct vcache *avc, int *slept) {
30     *slept = 0;
31
32     if (!VREFCOUNT_GT(avc, 0) && avc->opens == 0 &&
33         (avc->f.states & CUnlinkedDel) == 0) {
34
35         vnode_t tvp = AFSTOV(avc);
36         /* VREFCOUNT_GT only sees usecounts, not iocounts */
37         /* so this may fail to actually recycle the vnode now */
38         /* must call vnode_get to avoid races. */
39         if (vnode_get(tvp) == 0) {
40             *slept=1;
41             /* must release lock, since vnode_put will immediately
42                reclaim if there are no other users */
43             ReleaseWriteLock(&afs_xvcache);
44             AFS_GUNLOCK();
45             vnode_recycle(tvp);
46             vnode_put(tvp);
47             AFS_GLOCK();
48             ObtainWriteLock(&afs_xvcache, 336);
49         }
50         /* we can't use the vnode_recycle return value to figure
51          * this out, since the iocount we have to hold makes it
52          * always "fail" */
53         if (AFSTOV(avc) == tvp) {
54             if (*slept) {
55                 QRemove(&avc->vlruq);
56                 QAdd(&VLRU, &avc->vlruq);
57             }
58             return 0;
59         } else
60             return 1;
61     }
62     return 0;
63 }
64 #else
65 int
66 osi_TryEvictVCache(struct vcache *avc, int *slept) {
67     if (!VREFCOUNT_GT(avc,0)
68         || ((VREFCOUNT(avc) == 1) && (UBCINFOEXISTS(AFSTOV(avc))))
69         && avc->opens == 0 && (avc->f.states & CUnlinkedDel) == 0)
70     {
71         /*
72          * vgone() reclaims the vnode, which calls afs_FlushVCache(),
73          * then it puts the vnode on the free list.
74          * If we don't do this we end up with a cleaned vnode that's
75          * not on the free list.
76          */
77         AFS_GUNLOCK();
78         vgone(AFSTOV(avc));
79         AFS_GLOCK();
80         return 1;
81     }
82     return 0;
83 }
84 #endif /* AFS_DARWIN80_ENV */
85
86 void
87 osi_PrePopulateVCache(struct vcache *avc) {
88     memset(avc, 0, sizeof(struct vcache));
89
90     /* PPC Darwin 80 seems to be a BOZONLOCK environment, so we need this
91      * here ... */
92 #if defined(AFS_BOZONLOCK_ENV)
93     afs_BozonInit(&avc->pvnLock, avc);
94 #endif
95 }
96
97 void
98 osi_AttachVnode(struct vcache *avc, int seq) {
99     ReleaseWriteLock(&afs_xvcache);
100     AFS_GUNLOCK();
101     afs_darwin_getnewvnode(avc, seq ? 0 : 1);  /* includes one refcount */
102     AFS_GLOCK();
103     ObtainWriteLock(&afs_xvcache,338);
104 #ifdef AFS_DARWIN80_ENV
105     LOCKINIT(avc->rwlock);
106 #else
107     lockinit(&avc->rwlock, PINOD, "vcache", 0, 0);
108 #endif
109 }
110
111 void
112 osi_PostPopulateVCache(struct vcache *avc) {
113 #if !defined(AFS_DARWIN80_ENV)
114    avc->v->v_mount = afs_globalVFS;
115 #endif
116    vSetType(avc, VREG);
117 }