diff options
Diffstat (limited to 'fs/nfs/inode.c')
| -rw-r--r-- | fs/nfs/inode.c | 71 |
1 files changed, 61 insertions, 10 deletions
diff --git a/fs/nfs/inode.c b/fs/nfs/inode.c index 2c23d067e2a6..df23f987da6b 100644 --- a/fs/nfs/inode.c +++ b/fs/nfs/inode.c | |||
| @@ -370,7 +370,6 @@ nfs_setattr(struct dentry *dentry, struct iattr *attr) | |||
| 370 | if ((attr->ia_valid & ~ATTR_FILE) == 0) | 370 | if ((attr->ia_valid & ~ATTR_FILE) == 0) |
| 371 | return 0; | 371 | return 0; |
| 372 | 372 | ||
| 373 | lock_kernel(); | ||
| 374 | /* Write all dirty data */ | 373 | /* Write all dirty data */ |
| 375 | if (S_ISREG(inode->i_mode)) { | 374 | if (S_ISREG(inode->i_mode)) { |
| 376 | filemap_write_and_wait(inode->i_mapping); | 375 | filemap_write_and_wait(inode->i_mapping); |
| @@ -384,11 +383,66 @@ nfs_setattr(struct dentry *dentry, struct iattr *attr) | |||
| 384 | error = NFS_PROTO(inode)->setattr(dentry, &fattr, attr); | 383 | error = NFS_PROTO(inode)->setattr(dentry, &fattr, attr); |
| 385 | if (error == 0) | 384 | if (error == 0) |
| 386 | nfs_refresh_inode(inode, &fattr); | 385 | nfs_refresh_inode(inode, &fattr); |
| 387 | unlock_kernel(); | ||
| 388 | return error; | 386 | return error; |
| 389 | } | 387 | } |
| 390 | 388 | ||
| 391 | /** | 389 | /** |
| 390 | * nfs_vmtruncate - unmap mappings "freed" by truncate() syscall | ||
| 391 | * @inode: inode of the file used | ||
| 392 | * @offset: file offset to start truncating | ||
| 393 | * | ||
| 394 | * This is a copy of the common vmtruncate, but with the locking | ||
| 395 | * corrected to take into account the fact that NFS requires | ||
| 396 | * inode->i_size to be updated under the inode->i_lock. | ||
| 397 | */ | ||
| 398 | static int nfs_vmtruncate(struct inode * inode, loff_t offset) | ||
| 399 | { | ||
| 400 | if (i_size_read(inode) < offset) { | ||
| 401 | unsigned long limit; | ||
| 402 | |||
| 403 | limit = current->signal->rlim[RLIMIT_FSIZE].rlim_cur; | ||
| 404 | if (limit != RLIM_INFINITY && offset > limit) | ||
| 405 | goto out_sig; | ||
| 406 | if (offset > inode->i_sb->s_maxbytes) | ||
| 407 | goto out_big; | ||
| 408 | spin_lock(&inode->i_lock); | ||
| 409 | i_size_write(inode, offset); | ||
| 410 | spin_unlock(&inode->i_lock); | ||
| 411 | } else { | ||
| 412 | struct address_space *mapping = inode->i_mapping; | ||
| 413 | |||
| 414 | /* | ||
| 415 | * truncation of in-use swapfiles is disallowed - it would | ||
| 416 | * cause subsequent swapout to scribble on the now-freed | ||
| 417 | * blocks. | ||
| 418 | */ | ||
| 419 | if (IS_SWAPFILE(inode)) | ||
| 420 | return -ETXTBSY; | ||
| 421 | spin_lock(&inode->i_lock); | ||
| 422 | i_size_write(inode, offset); | ||
| 423 | spin_unlock(&inode->i_lock); | ||
| 424 | |||
| 425 | /* | ||
| 426 | * unmap_mapping_range is called twice, first simply for | ||
| 427 | * efficiency so that truncate_inode_pages does fewer | ||
| 428 | * single-page unmaps. However after this first call, and | ||
| 429 | * before truncate_inode_pages finishes, it is possible for | ||
| 430 | * private pages to be COWed, which remain after | ||
| 431 | * truncate_inode_pages finishes, hence the second | ||
| 432 | * unmap_mapping_range call must be made for correctness. | ||
| 433 | */ | ||
| 434 | unmap_mapping_range(mapping, offset + PAGE_SIZE - 1, 0, 1); | ||
| 435 | truncate_inode_pages(mapping, offset); | ||
| 436 | unmap_mapping_range(mapping, offset + PAGE_SIZE - 1, 0, 1); | ||
| 437 | } | ||
| 438 | return 0; | ||
| 439 | out_sig: | ||
| 440 | send_sig(SIGXFSZ, current, 0); | ||
| 441 | out_big: | ||
| 442 | return -EFBIG; | ||
| 443 | } | ||
| 444 | |||
| 445 | /** | ||
| 392 | * nfs_setattr_update_inode - Update inode metadata after a setattr call. | 446 | * nfs_setattr_update_inode - Update inode metadata after a setattr call. |
| 393 | * @inode: pointer to struct inode | 447 | * @inode: pointer to struct inode |
| 394 | * @attr: pointer to struct iattr | 448 | * @attr: pointer to struct iattr |
| @@ -414,8 +468,7 @@ void nfs_setattr_update_inode(struct inode *inode, struct iattr *attr) | |||
| 414 | } | 468 | } |
| 415 | if ((attr->ia_valid & ATTR_SIZE) != 0) { | 469 | if ((attr->ia_valid & ATTR_SIZE) != 0) { |
| 416 | nfs_inc_stats(inode, NFSIOS_SETATTRTRUNC); | 470 | nfs_inc_stats(inode, NFSIOS_SETATTRTRUNC); |
| 417 | inode->i_size = attr->ia_size; | 471 | nfs_vmtruncate(inode, attr->ia_size); |
| 418 | vmtruncate(inode, attr->ia_size); | ||
| 419 | } | 472 | } |
| 420 | } | 473 | } |
| 421 | 474 | ||
| @@ -645,7 +698,6 @@ __nfs_revalidate_inode(struct nfs_server *server, struct inode *inode) | |||
| 645 | inode->i_sb->s_id, (long long)NFS_FILEID(inode)); | 698 | inode->i_sb->s_id, (long long)NFS_FILEID(inode)); |
| 646 | 699 | ||
| 647 | nfs_inc_stats(inode, NFSIOS_INODEREVALIDATE); | 700 | nfs_inc_stats(inode, NFSIOS_INODEREVALIDATE); |
| 648 | lock_kernel(); | ||
| 649 | if (is_bad_inode(inode)) | 701 | if (is_bad_inode(inode)) |
| 650 | goto out_nowait; | 702 | goto out_nowait; |
| 651 | if (NFS_STALE(inode)) | 703 | if (NFS_STALE(inode)) |
| @@ -694,7 +746,6 @@ __nfs_revalidate_inode(struct nfs_server *server, struct inode *inode) | |||
| 694 | nfs_wake_up_inode(inode); | 746 | nfs_wake_up_inode(inode); |
| 695 | 747 | ||
| 696 | out_nowait: | 748 | out_nowait: |
| 697 | unlock_kernel(); | ||
| 698 | return status; | 749 | return status; |
| 699 | } | 750 | } |
| 700 | 751 | ||
| @@ -829,9 +880,9 @@ static void nfs_wcc_update_inode(struct inode *inode, struct nfs_fattr *fattr) | |||
| 829 | if (S_ISDIR(inode->i_mode)) | 880 | if (S_ISDIR(inode->i_mode)) |
| 830 | nfsi->cache_validity |= NFS_INO_INVALID_DATA; | 881 | nfsi->cache_validity |= NFS_INO_INVALID_DATA; |
| 831 | } | 882 | } |
| 832 | if (inode->i_size == nfs_size_to_loff_t(fattr->pre_size) && | 883 | if (i_size_read(inode) == nfs_size_to_loff_t(fattr->pre_size) && |
| 833 | nfsi->npages == 0) | 884 | nfsi->npages == 0) |
| 834 | inode->i_size = nfs_size_to_loff_t(fattr->size); | 885 | i_size_write(inode, nfs_size_to_loff_t(fattr->size)); |
| 835 | } | 886 | } |
| 836 | } | 887 | } |
| 837 | 888 | ||
| @@ -972,7 +1023,7 @@ int nfs_post_op_update_inode_force_wcc(struct inode *inode, struct nfs_fattr *fa | |||
| 972 | (fattr->valid & NFS_ATTR_WCC) == 0) { | 1023 | (fattr->valid & NFS_ATTR_WCC) == 0) { |
| 973 | memcpy(&fattr->pre_ctime, &inode->i_ctime, sizeof(fattr->pre_ctime)); | 1024 | memcpy(&fattr->pre_ctime, &inode->i_ctime, sizeof(fattr->pre_ctime)); |
| 974 | memcpy(&fattr->pre_mtime, &inode->i_mtime, sizeof(fattr->pre_mtime)); | 1025 | memcpy(&fattr->pre_mtime, &inode->i_mtime, sizeof(fattr->pre_mtime)); |
| 975 | fattr->pre_size = inode->i_size; | 1026 | fattr->pre_size = i_size_read(inode); |
| 976 | fattr->valid |= NFS_ATTR_WCC; | 1027 | fattr->valid |= NFS_ATTR_WCC; |
| 977 | } | 1028 | } |
| 978 | return nfs_post_op_update_inode(inode, fattr); | 1029 | return nfs_post_op_update_inode(inode, fattr); |
| @@ -1057,7 +1108,7 @@ static int nfs_update_inode(struct inode *inode, struct nfs_fattr *fattr) | |||
| 1057 | /* Do we perhaps have any outstanding writes, or has | 1108 | /* Do we perhaps have any outstanding writes, or has |
| 1058 | * the file grown beyond our last write? */ | 1109 | * the file grown beyond our last write? */ |
| 1059 | if (nfsi->npages == 0 || new_isize > cur_isize) { | 1110 | if (nfsi->npages == 0 || new_isize > cur_isize) { |
| 1060 | inode->i_size = new_isize; | 1111 | i_size_write(inode, new_isize); |
| 1061 | invalid |= NFS_INO_INVALID_ATTR|NFS_INO_INVALID_DATA; | 1112 | invalid |= NFS_INO_INVALID_ATTR|NFS_INO_INVALID_DATA; |
| 1062 | } | 1113 | } |
| 1063 | dprintk("NFS: isize change on server for file %s/%ld\n", | 1114 | dprintk("NFS: isize change on server for file %s/%ld\n", |
