diff options
author | Trond Myklebust <Trond.Myklebust@netapp.com> | 2012-09-20 17:31:43 -0400 |
---|---|---|
committer | Trond Myklebust <Trond.Myklebust@netapp.com> | 2012-09-28 16:03:14 -0400 |
commit | 9c6263819f25254f2ed48f076b50096dd5893dfb (patch) | |
tree | d95527cbeba0d44973f286602a35ff2c21c811bd /fs/nfs | |
parent | 6622c3ea059b2fed49924b74db41d1e0f065fbd3 (diff) |
NFSv4.1: Clean up the removal of pnfs_layout_hdr from the server list
Move the code into pnfs_free_layout_hdr(), and add checks to
get_layout_by_fh_locked to ensure that they don't reference a layout
that is being freed.
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Diffstat (limited to 'fs/nfs')
-rw-r--r-- | fs/nfs/callback_proc.c | 19 | ||||
-rw-r--r-- | fs/nfs/pnfs.c | 29 |
2 files changed, 28 insertions, 20 deletions
diff --git a/fs/nfs/callback_proc.c b/fs/nfs/callback_proc.c index 24252fea2c9..76b4a7a3e55 100644 --- a/fs/nfs/callback_proc.c +++ b/fs/nfs/callback_proc.c | |||
@@ -122,7 +122,15 @@ static struct pnfs_layout_hdr * get_layout_by_fh_locked(struct nfs_client *clp, | |||
122 | ino = igrab(lo->plh_inode); | 122 | ino = igrab(lo->plh_inode); |
123 | if (!ino) | 123 | if (!ino) |
124 | continue; | 124 | continue; |
125 | spin_lock(&ino->i_lock); | ||
126 | /* Is this layout in the process of being freed? */ | ||
127 | if (NFS_I(ino)->layout != lo) { | ||
128 | spin_unlock(&ino->i_lock); | ||
129 | iput(ino); | ||
130 | continue; | ||
131 | } | ||
125 | pnfs_get_layout_hdr(lo); | 132 | pnfs_get_layout_hdr(lo); |
133 | spin_unlock(&ino->i_lock); | ||
126 | return lo; | 134 | return lo; |
127 | } | 135 | } |
128 | } | 136 | } |
@@ -196,9 +204,18 @@ static u32 initiate_bulk_draining(struct nfs_client *clp, | |||
196 | continue; | 204 | continue; |
197 | 205 | ||
198 | list_for_each_entry(lo, &server->layouts, plh_layouts) { | 206 | list_for_each_entry(lo, &server->layouts, plh_layouts) { |
199 | if (!igrab(lo->plh_inode)) | 207 | ino = igrab(lo->plh_inode); |
208 | if (ino) | ||
209 | continue; | ||
210 | spin_lock(&ino->i_lock); | ||
211 | /* Is this layout in the process of being freed? */ | ||
212 | if (NFS_I(ino)->layout != lo) { | ||
213 | spin_unlock(&ino->i_lock); | ||
214 | iput(ino); | ||
200 | continue; | 215 | continue; |
216 | } | ||
201 | pnfs_get_layout_hdr(lo); | 217 | pnfs_get_layout_hdr(lo); |
218 | spin_unlock(&ino->i_lock); | ||
202 | BUG_ON(!list_empty(&lo->plh_bulk_recall)); | 219 | BUG_ON(!list_empty(&lo->plh_bulk_recall)); |
203 | list_add(&lo->plh_bulk_recall, &recall_list); | 220 | list_add(&lo->plh_bulk_recall, &recall_list); |
204 | } | 221 | } |
diff --git a/fs/nfs/pnfs.c b/fs/nfs/pnfs.c index 08663146f5f..11cc0ad6b40 100644 --- a/fs/nfs/pnfs.c +++ b/fs/nfs/pnfs.c | |||
@@ -207,7 +207,16 @@ pnfs_alloc_layout_hdr(struct inode *ino, gfp_t gfp_flags) | |||
207 | static void | 207 | static void |
208 | pnfs_free_layout_hdr(struct pnfs_layout_hdr *lo) | 208 | pnfs_free_layout_hdr(struct pnfs_layout_hdr *lo) |
209 | { | 209 | { |
210 | struct pnfs_layoutdriver_type *ld = NFS_SERVER(lo->plh_inode)->pnfs_curr_ld; | 210 | struct nfs_server *server = NFS_SERVER(lo->plh_inode); |
211 | struct pnfs_layoutdriver_type *ld = server->pnfs_curr_ld; | ||
212 | |||
213 | if (!list_empty(&lo->plh_layouts)) { | ||
214 | struct nfs_client *clp = server->nfs_client; | ||
215 | |||
216 | spin_lock(&clp->cl_lock); | ||
217 | list_del_init(&lo->plh_layouts); | ||
218 | spin_unlock(&clp->cl_lock); | ||
219 | } | ||
211 | put_rpccred(lo->plh_lc_cred); | 220 | put_rpccred(lo->plh_lc_cred); |
212 | return ld->alloc_layout_hdr ? ld->free_layout_hdr(lo) : kfree(lo); | 221 | return ld->alloc_layout_hdr ? ld->free_layout_hdr(lo) : kfree(lo); |
213 | } | 222 | } |
@@ -217,7 +226,6 @@ pnfs_detach_layout_hdr(struct pnfs_layout_hdr *lo) | |||
217 | { | 226 | { |
218 | struct nfs_inode *nfsi = NFS_I(lo->plh_inode); | 227 | struct nfs_inode *nfsi = NFS_I(lo->plh_inode); |
219 | dprintk("%s: freeing layout cache %p\n", __func__, lo); | 228 | dprintk("%s: freeing layout cache %p\n", __func__, lo); |
220 | BUG_ON(!list_empty(&lo->plh_layouts)); | ||
221 | nfsi->layout = NULL; | 229 | nfsi->layout = NULL; |
222 | /* Reset MDS Threshold I/O counters */ | 230 | /* Reset MDS Threshold I/O counters */ |
223 | nfsi->write_io = 0; | 231 | nfsi->write_io = 0; |
@@ -480,22 +488,10 @@ void | |||
480 | pnfs_free_lseg_list(struct list_head *free_me) | 488 | pnfs_free_lseg_list(struct list_head *free_me) |
481 | { | 489 | { |
482 | struct pnfs_layout_segment *lseg, *tmp; | 490 | struct pnfs_layout_segment *lseg, *tmp; |
483 | struct pnfs_layout_hdr *lo; | ||
484 | 491 | ||
485 | if (list_empty(free_me)) | 492 | if (list_empty(free_me)) |
486 | return; | 493 | return; |
487 | 494 | ||
488 | lo = list_first_entry(free_me, struct pnfs_layout_segment, | ||
489 | pls_list)->pls_layout; | ||
490 | |||
491 | if (test_bit(NFS_LAYOUT_DESTROYED, &lo->plh_flags)) { | ||
492 | struct nfs_client *clp; | ||
493 | |||
494 | clp = NFS_SERVER(lo->plh_inode)->nfs_client; | ||
495 | spin_lock(&clp->cl_lock); | ||
496 | list_del_init(&lo->plh_layouts); | ||
497 | spin_unlock(&clp->cl_lock); | ||
498 | } | ||
499 | list_for_each_entry_safe(lseg, tmp, free_me, pls_list) { | 495 | list_for_each_entry_safe(lseg, tmp, free_me, pls_list) { |
500 | list_del(&lseg->pls_list); | 496 | list_del(&lseg->pls_list); |
501 | free_lseg(lseg); | 497 | free_lseg(lseg); |
@@ -1148,11 +1144,6 @@ pnfs_update_layout(struct inode *ino, | |||
1148 | arg.length = PAGE_CACHE_ALIGN(arg.length); | 1144 | arg.length = PAGE_CACHE_ALIGN(arg.length); |
1149 | 1145 | ||
1150 | lseg = send_layoutget(lo, ctx, &arg, gfp_flags); | 1146 | lseg = send_layoutget(lo, ctx, &arg, gfp_flags); |
1151 | if (!lseg && first) { | ||
1152 | spin_lock(&clp->cl_lock); | ||
1153 | list_del_init(&lo->plh_layouts); | ||
1154 | spin_unlock(&clp->cl_lock); | ||
1155 | } | ||
1156 | atomic_dec(&lo->plh_outstanding); | 1147 | atomic_dec(&lo->plh_outstanding); |
1157 | out_put_layout_hdr: | 1148 | out_put_layout_hdr: |
1158 | pnfs_put_layout_hdr(lo); | 1149 | pnfs_put_layout_hdr(lo); |