diff options
Diffstat (limited to 'fs/nfsd/nfscache.c')
-rw-r--r-- | fs/nfsd/nfscache.c | 36 |
1 files changed, 27 insertions, 9 deletions
diff --git a/fs/nfsd/nfscache.c b/fs/nfsd/nfscache.c index 4aad9e4a6161..363bc6188155 100644 --- a/fs/nfsd/nfscache.c +++ b/fs/nfsd/nfscache.c | |||
@@ -47,10 +47,34 @@ static int nfsd_cache_append(struct svc_rqst *rqstp, struct kvec *vec); | |||
47 | */ | 47 | */ |
48 | static DEFINE_SPINLOCK(cache_lock); | 48 | static DEFINE_SPINLOCK(cache_lock); |
49 | 49 | ||
50 | int nfsd_reply_cache_init(void) | 50 | static struct svc_cacherep * |
51 | nfsd_reply_cache_alloc(void) | ||
51 | { | 52 | { |
52 | struct svc_cacherep *rp; | 53 | struct svc_cacherep *rp; |
54 | |||
55 | rp = kmem_cache_alloc(drc_slab, GFP_KERNEL); | ||
56 | if (rp) { | ||
57 | rp->c_state = RC_UNUSED; | ||
58 | rp->c_type = RC_NOCACHE; | ||
59 | INIT_LIST_HEAD(&rp->c_lru); | ||
60 | INIT_HLIST_NODE(&rp->c_hash); | ||
61 | } | ||
62 | return rp; | ||
63 | } | ||
64 | |||
65 | static void | ||
66 | nfsd_reply_cache_free_locked(struct svc_cacherep *rp) | ||
67 | { | ||
68 | if (rp->c_state == RC_DONE && rp->c_type == RC_REPLBUFF) | ||
69 | kfree(rp->c_replvec.iov_base); | ||
70 | list_del(&rp->c_lru); | ||
71 | kmem_cache_free(drc_slab, rp); | ||
72 | } | ||
73 | |||
74 | int nfsd_reply_cache_init(void) | ||
75 | { | ||
53 | int i; | 76 | int i; |
77 | struct svc_cacherep *rp; | ||
54 | 78 | ||
55 | drc_slab = kmem_cache_create("nfsd_drc", sizeof(struct svc_cacherep), | 79 | drc_slab = kmem_cache_create("nfsd_drc", sizeof(struct svc_cacherep), |
56 | 0, 0, NULL); | 80 | 0, 0, NULL); |
@@ -60,13 +84,10 @@ int nfsd_reply_cache_init(void) | |||
60 | INIT_LIST_HEAD(&lru_head); | 84 | INIT_LIST_HEAD(&lru_head); |
61 | i = CACHESIZE; | 85 | i = CACHESIZE; |
62 | while (i) { | 86 | while (i) { |
63 | rp = kmem_cache_alloc(drc_slab, GFP_KERNEL); | 87 | rp = nfsd_reply_cache_alloc(); |
64 | if (!rp) | 88 | if (!rp) |
65 | goto out_nomem; | 89 | goto out_nomem; |
66 | list_add(&rp->c_lru, &lru_head); | 90 | list_add(&rp->c_lru, &lru_head); |
67 | rp->c_state = RC_UNUSED; | ||
68 | rp->c_type = RC_NOCACHE; | ||
69 | INIT_HLIST_NODE(&rp->c_hash); | ||
70 | i--; | 91 | i--; |
71 | } | 92 | } |
72 | 93 | ||
@@ -88,10 +109,7 @@ void nfsd_reply_cache_shutdown(void) | |||
88 | 109 | ||
89 | while (!list_empty(&lru_head)) { | 110 | while (!list_empty(&lru_head)) { |
90 | rp = list_entry(lru_head.next, struct svc_cacherep, c_lru); | 111 | rp = list_entry(lru_head.next, struct svc_cacherep, c_lru); |
91 | if (rp->c_state == RC_DONE && rp->c_type == RC_REPLBUFF) | 112 | nfsd_reply_cache_free_locked(rp); |
92 | kfree(rp->c_replvec.iov_base); | ||
93 | list_del(&rp->c_lru); | ||
94 | kmem_cache_free(drc_slab, rp); | ||
95 | } | 113 | } |
96 | 114 | ||
97 | cache_disabled = 1; | 115 | cache_disabled = 1; |