diff options
author | Trond Myklebust <Trond.Myklebust@netapp.com> | 2012-09-20 20:57:11 -0400 |
---|---|---|
committer | Trond Myklebust <Trond.Myklebust@netapp.com> | 2012-09-28 16:03:15 -0400 |
commit | 8f0d27dc5d77b084b2e2fe6d883c4d5776287842 (patch) | |
tree | b3dcc9cb6076fca8204288535cf1789b108a71fc /fs | |
parent | 905ca191cfe1ab18822d86e3ddef1b1b38832fdc (diff) |
NFSv4.1: Balance pnfs_layout_hdr refcount in pnfs_layout_(insert|remove)_lseg
Ensure that the reference count for pnfs_layout_hdr reverts to the
original value after a call to pnfs_layout_remove_lseg().
Note that the caller is expected to hold a reference to the struct
pnfs_layout_hdr.
Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Diffstat (limited to 'fs')
-rw-r--r-- | fs/nfs/pnfs.c | 6 |
1 files changed, 4 insertions, 2 deletions
diff --git a/fs/nfs/pnfs.c b/fs/nfs/pnfs.c index c34ba9a0a46..bdd93b96905 100644 --- a/fs/nfs/pnfs.c +++ b/fs/nfs/pnfs.c | |||
@@ -320,8 +320,6 @@ static void pnfs_free_lseg(struct pnfs_layout_segment *lseg) | |||
320 | struct inode *ino = lseg->pls_layout->plh_inode; | 320 | struct inode *ino = lseg->pls_layout->plh_inode; |
321 | 321 | ||
322 | NFS_SERVER(ino)->pnfs_curr_ld->free_lseg(lseg); | 322 | NFS_SERVER(ino)->pnfs_curr_ld->free_lseg(lseg); |
323 | /* Matched by pnfs_get_layout_hdr in pnfs_layout_insert_lseg */ | ||
324 | pnfs_put_layout_hdr(NFS_I(ino)->layout); | ||
325 | } | 323 | } |
326 | 324 | ||
327 | static void | 325 | static void |
@@ -332,6 +330,8 @@ pnfs_layout_remove_lseg(struct pnfs_layout_hdr *lo, | |||
332 | 330 | ||
333 | WARN_ON(test_bit(NFS_LSEG_VALID, &lseg->pls_flags)); | 331 | WARN_ON(test_bit(NFS_LSEG_VALID, &lseg->pls_flags)); |
334 | list_del_init(&lseg->pls_list); | 332 | list_del_init(&lseg->pls_list); |
333 | /* Matched by pnfs_get_layout_hdr in pnfs_layout_insert_lseg */ | ||
334 | atomic_dec(&lo->plh_refcount); | ||
335 | if (list_empty(&lo->plh_segs)) | 335 | if (list_empty(&lo->plh_segs)) |
336 | set_bit(NFS_LAYOUT_DESTROYED, &lo->plh_flags); | 336 | set_bit(NFS_LAYOUT_DESTROYED, &lo->plh_flags); |
337 | rpc_wake_up(&NFS_SERVER(inode)->roc_rpcwaitq); | 337 | rpc_wake_up(&NFS_SERVER(inode)->roc_rpcwaitq); |
@@ -352,9 +352,11 @@ pnfs_put_lseg(struct pnfs_layout_segment *lseg) | |||
352 | lo = lseg->pls_layout; | 352 | lo = lseg->pls_layout; |
353 | inode = lo->plh_inode; | 353 | inode = lo->plh_inode; |
354 | if (atomic_dec_and_lock(&lseg->pls_refcount, &inode->i_lock)) { | 354 | if (atomic_dec_and_lock(&lseg->pls_refcount, &inode->i_lock)) { |
355 | pnfs_get_layout_hdr(lo); | ||
355 | pnfs_layout_remove_lseg(lo, lseg); | 356 | pnfs_layout_remove_lseg(lo, lseg); |
356 | spin_unlock(&inode->i_lock); | 357 | spin_unlock(&inode->i_lock); |
357 | pnfs_free_lseg(lseg); | 358 | pnfs_free_lseg(lseg); |
359 | pnfs_put_layout_hdr(lo); | ||
358 | } | 360 | } |
359 | } | 361 | } |
360 | EXPORT_SYMBOL_GPL(pnfs_put_lseg); | 362 | EXPORT_SYMBOL_GPL(pnfs_put_lseg); |