diff options
Diffstat (limited to 'fs/cifs/file.c')
-rw-r--r-- | fs/cifs/file.c | 79 |
1 files changed, 48 insertions, 31 deletions
diff --git a/fs/cifs/file.c b/fs/cifs/file.c index 14a1c72ced92..77c990f0cb98 100644 --- a/fs/cifs/file.c +++ b/fs/cifs/file.c | |||
@@ -127,8 +127,7 @@ static inline int cifs_open_inode_helper(struct inode *inode, struct file *file, | |||
127 | if (file->f_dentry->d_inode->i_mapping) { | 127 | if (file->f_dentry->d_inode->i_mapping) { |
128 | /* BB no need to lock inode until after invalidate | 128 | /* BB no need to lock inode until after invalidate |
129 | since namei code should already have it locked? */ | 129 | since namei code should already have it locked? */ |
130 | filemap_fdatawrite(file->f_dentry->d_inode->i_mapping); | 130 | filemap_write_and_wait(file->f_dentry->d_inode->i_mapping); |
131 | filemap_fdatawait(file->f_dentry->d_inode->i_mapping); | ||
132 | } | 131 | } |
133 | cFYI(1, ("invalidating remote inode since open detected it " | 132 | cFYI(1, ("invalidating remote inode since open detected it " |
134 | "changed")); | 133 | "changed")); |
@@ -419,8 +418,7 @@ static int cifs_reopen_file(struct inode *inode, struct file *file, | |||
419 | pCifsInode = CIFS_I(inode); | 418 | pCifsInode = CIFS_I(inode); |
420 | if (pCifsInode) { | 419 | if (pCifsInode) { |
421 | if (can_flush) { | 420 | if (can_flush) { |
422 | filemap_fdatawrite(inode->i_mapping); | 421 | filemap_write_and_wait(inode->i_mapping); |
423 | filemap_fdatawait(inode->i_mapping); | ||
424 | /* temporarily disable caching while we | 422 | /* temporarily disable caching while we |
425 | go to server to get inode info */ | 423 | go to server to get inode info */ |
426 | pCifsInode->clientCanCacheAll = FALSE; | 424 | pCifsInode->clientCanCacheAll = FALSE; |
@@ -555,13 +553,13 @@ int cifs_closedir(struct inode *inode, struct file *file) | |||
555 | } | 553 | } |
556 | ptmp = pCFileStruct->srch_inf.ntwrk_buf_start; | 554 | ptmp = pCFileStruct->srch_inf.ntwrk_buf_start; |
557 | if (ptmp) { | 555 | if (ptmp) { |
558 | /* BB removeme BB */ cFYI(1, ("freeing smb buf in srch struct in closedir")); | 556 | cFYI(1, ("closedir free smb buf in srch struct")); |
559 | pCFileStruct->srch_inf.ntwrk_buf_start = NULL; | 557 | pCFileStruct->srch_inf.ntwrk_buf_start = NULL; |
560 | cifs_buf_release(ptmp); | 558 | cifs_buf_release(ptmp); |
561 | } | 559 | } |
562 | ptmp = pCFileStruct->search_resume_name; | 560 | ptmp = pCFileStruct->search_resume_name; |
563 | if (ptmp) { | 561 | if (ptmp) { |
564 | /* BB removeme BB */ cFYI(1, ("freeing resume name in closedir")); | 562 | cFYI(1, ("closedir free resume name")); |
565 | pCFileStruct->search_resume_name = NULL; | 563 | pCFileStruct->search_resume_name = NULL; |
566 | kfree(ptmp); | 564 | kfree(ptmp); |
567 | } | 565 | } |
@@ -870,10 +868,9 @@ static ssize_t cifs_write(struct file *file, const char *write_data, | |||
870 | if (rc != 0) | 868 | if (rc != 0) |
871 | break; | 869 | break; |
872 | } | 870 | } |
873 | #ifdef CONFIG_CIFS_EXPERIMENTAL | ||
874 | /* BB FIXME We can not sign across two buffers yet */ | 871 | /* BB FIXME We can not sign across two buffers yet */ |
875 | if((experimEnabled) && ((pTcon->ses->server->secMode & | 872 | if((pTcon->ses->server->secMode & |
876 | (SECMODE_SIGN_REQUIRED | SECMODE_SIGN_ENABLED)) == 0)) { | 873 | (SECMODE_SIGN_REQUIRED | SECMODE_SIGN_ENABLED)) == 0) { |
877 | struct kvec iov[2]; | 874 | struct kvec iov[2]; |
878 | unsigned int len; | 875 | unsigned int len; |
879 | 876 | ||
@@ -889,7 +886,6 @@ static ssize_t cifs_write(struct file *file, const char *write_data, | |||
889 | iov, 1, long_op); | 886 | iov, 1, long_op); |
890 | } else | 887 | } else |
891 | /* BB FIXME fixup indentation of line below */ | 888 | /* BB FIXME fixup indentation of line below */ |
892 | #endif | ||
893 | rc = CIFSSMBWrite(xid, pTcon, | 889 | rc = CIFSSMBWrite(xid, pTcon, |
894 | open_file->netfid, | 890 | open_file->netfid, |
895 | min_t(const int, cifs_sb->wsize, | 891 | min_t(const int, cifs_sb->wsize, |
@@ -1026,7 +1022,6 @@ static int cifs_partialpagewrite(struct page *page, unsigned from, unsigned to) | |||
1026 | return rc; | 1022 | return rc; |
1027 | } | 1023 | } |
1028 | 1024 | ||
1029 | #ifdef CONFIG_CIFS_EXPERIMENTAL | ||
1030 | static int cifs_writepages(struct address_space *mapping, | 1025 | static int cifs_writepages(struct address_space *mapping, |
1031 | struct writeback_control *wbc) | 1026 | struct writeback_control *wbc) |
1032 | { | 1027 | { |
@@ -1229,7 +1224,6 @@ retry: | |||
1229 | 1224 | ||
1230 | return rc; | 1225 | return rc; |
1231 | } | 1226 | } |
1232 | #endif | ||
1233 | 1227 | ||
1234 | static int cifs_writepage(struct page* page, struct writeback_control *wbc) | 1228 | static int cifs_writepage(struct page* page, struct writeback_control *wbc) |
1235 | { | 1229 | { |
@@ -1428,6 +1422,7 @@ ssize_t cifs_user_read(struct file *file, char __user *read_data, | |||
1428 | rc = -EAGAIN; | 1422 | rc = -EAGAIN; |
1429 | smb_read_data = NULL; | 1423 | smb_read_data = NULL; |
1430 | while (rc == -EAGAIN) { | 1424 | while (rc == -EAGAIN) { |
1425 | int buf_type = CIFS_NO_BUFFER; | ||
1431 | if ((open_file->invalidHandle) && | 1426 | if ((open_file->invalidHandle) && |
1432 | (!open_file->closePend)) { | 1427 | (!open_file->closePend)) { |
1433 | rc = cifs_reopen_file(file->f_dentry->d_inode, | 1428 | rc = cifs_reopen_file(file->f_dentry->d_inode, |
@@ -1436,20 +1431,22 @@ ssize_t cifs_user_read(struct file *file, char __user *read_data, | |||
1436 | break; | 1431 | break; |
1437 | } | 1432 | } |
1438 | rc = CIFSSMBRead(xid, pTcon, | 1433 | rc = CIFSSMBRead(xid, pTcon, |
1439 | open_file->netfid, | 1434 | open_file->netfid, |
1440 | current_read_size, *poffset, | 1435 | current_read_size, *poffset, |
1441 | &bytes_read, &smb_read_data); | 1436 | &bytes_read, &smb_read_data, |
1437 | &buf_type); | ||
1442 | pSMBr = (struct smb_com_read_rsp *)smb_read_data; | 1438 | pSMBr = (struct smb_com_read_rsp *)smb_read_data; |
1443 | if (copy_to_user(current_offset, | 1439 | if (copy_to_user(current_offset, |
1444 | smb_read_data + 4 /* RFC1001 hdr */ | 1440 | smb_read_data + 4 /* RFC1001 hdr */ |
1445 | + le16_to_cpu(pSMBr->DataOffset), | 1441 | + le16_to_cpu(pSMBr->DataOffset), |
1446 | bytes_read)) { | 1442 | bytes_read)) { |
1447 | rc = -EFAULT; | 1443 | rc = -EFAULT; |
1448 | FreeXid(xid); | 1444 | } |
1449 | return rc; | ||
1450 | } | ||
1451 | if (smb_read_data) { | 1445 | if (smb_read_data) { |
1452 | cifs_buf_release(smb_read_data); | 1446 | if(buf_type == CIFS_SMALL_BUFFER) |
1447 | cifs_small_buf_release(smb_read_data); | ||
1448 | else if(buf_type == CIFS_LARGE_BUFFER) | ||
1449 | cifs_buf_release(smb_read_data); | ||
1453 | smb_read_data = NULL; | 1450 | smb_read_data = NULL; |
1454 | } | 1451 | } |
1455 | } | 1452 | } |
@@ -1482,6 +1479,7 @@ static ssize_t cifs_read(struct file *file, char *read_data, size_t read_size, | |||
1482 | int xid; | 1479 | int xid; |
1483 | char *current_offset; | 1480 | char *current_offset; |
1484 | struct cifsFileInfo *open_file; | 1481 | struct cifsFileInfo *open_file; |
1482 | int buf_type = CIFS_NO_BUFFER; | ||
1485 | 1483 | ||
1486 | xid = GetXid(); | 1484 | xid = GetXid(); |
1487 | cifs_sb = CIFS_SB(file->f_dentry->d_sb); | 1485 | cifs_sb = CIFS_SB(file->f_dentry->d_sb); |
@@ -1518,9 +1516,10 @@ static ssize_t cifs_read(struct file *file, char *read_data, size_t read_size, | |||
1518 | break; | 1516 | break; |
1519 | } | 1517 | } |
1520 | rc = CIFSSMBRead(xid, pTcon, | 1518 | rc = CIFSSMBRead(xid, pTcon, |
1521 | open_file->netfid, | 1519 | open_file->netfid, |
1522 | current_read_size, *poffset, | 1520 | current_read_size, *poffset, |
1523 | &bytes_read, ¤t_offset); | 1521 | &bytes_read, ¤t_offset, |
1522 | &buf_type); | ||
1524 | } | 1523 | } |
1525 | if (rc || (bytes_read == 0)) { | 1524 | if (rc || (bytes_read == 0)) { |
1526 | if (total_read) { | 1525 | if (total_read) { |
@@ -1618,6 +1617,7 @@ static int cifs_readpages(struct file *file, struct address_space *mapping, | |||
1618 | struct smb_com_read_rsp *pSMBr; | 1617 | struct smb_com_read_rsp *pSMBr; |
1619 | struct pagevec lru_pvec; | 1618 | struct pagevec lru_pvec; |
1620 | struct cifsFileInfo *open_file; | 1619 | struct cifsFileInfo *open_file; |
1620 | int buf_type = CIFS_NO_BUFFER; | ||
1621 | 1621 | ||
1622 | xid = GetXid(); | 1622 | xid = GetXid(); |
1623 | if (file->private_data == NULL) { | 1623 | if (file->private_data == NULL) { |
@@ -1674,14 +1674,17 @@ static int cifs_readpages(struct file *file, struct address_space *mapping, | |||
1674 | } | 1674 | } |
1675 | 1675 | ||
1676 | rc = CIFSSMBRead(xid, pTcon, | 1676 | rc = CIFSSMBRead(xid, pTcon, |
1677 | open_file->netfid, | 1677 | open_file->netfid, |
1678 | read_size, offset, | 1678 | read_size, offset, |
1679 | &bytes_read, &smb_read_data); | 1679 | &bytes_read, &smb_read_data, |
1680 | 1680 | &buf_type); | |
1681 | /* BB more RC checks ? */ | 1681 | /* BB more RC checks ? */ |
1682 | if (rc== -EAGAIN) { | 1682 | if (rc== -EAGAIN) { |
1683 | if (smb_read_data) { | 1683 | if (smb_read_data) { |
1684 | cifs_buf_release(smb_read_data); | 1684 | if(buf_type == CIFS_SMALL_BUFFER) |
1685 | cifs_small_buf_release(smb_read_data); | ||
1686 | else if(buf_type == CIFS_LARGE_BUFFER) | ||
1687 | cifs_buf_release(smb_read_data); | ||
1685 | smb_read_data = NULL; | 1688 | smb_read_data = NULL; |
1686 | } | 1689 | } |
1687 | } | 1690 | } |
@@ -1738,7 +1741,10 @@ static int cifs_readpages(struct file *file, struct address_space *mapping, | |||
1738 | break; | 1741 | break; |
1739 | } | 1742 | } |
1740 | if (smb_read_data) { | 1743 | if (smb_read_data) { |
1741 | cifs_buf_release(smb_read_data); | 1744 | if(buf_type == CIFS_SMALL_BUFFER) |
1745 | cifs_small_buf_release(smb_read_data); | ||
1746 | else if(buf_type == CIFS_LARGE_BUFFER) | ||
1747 | cifs_buf_release(smb_read_data); | ||
1742 | smb_read_data = NULL; | 1748 | smb_read_data = NULL; |
1743 | } | 1749 | } |
1744 | bytes_read = 0; | 1750 | bytes_read = 0; |
@@ -1748,7 +1754,10 @@ static int cifs_readpages(struct file *file, struct address_space *mapping, | |||
1748 | 1754 | ||
1749 | /* need to free smb_read_data buf before exit */ | 1755 | /* need to free smb_read_data buf before exit */ |
1750 | if (smb_read_data) { | 1756 | if (smb_read_data) { |
1751 | cifs_buf_release(smb_read_data); | 1757 | if(buf_type == CIFS_SMALL_BUFFER) |
1758 | cifs_small_buf_release(smb_read_data); | ||
1759 | else if(buf_type == CIFS_LARGE_BUFFER) | ||
1760 | cifs_buf_release(smb_read_data); | ||
1752 | smb_read_data = NULL; | 1761 | smb_read_data = NULL; |
1753 | } | 1762 | } |
1754 | 1763 | ||
@@ -1827,10 +1836,20 @@ int is_size_safe_to_change(struct cifsInodeInfo *cifsInode) | |||
1827 | open_file = find_writable_file(cifsInode); | 1836 | open_file = find_writable_file(cifsInode); |
1828 | 1837 | ||
1829 | if(open_file) { | 1838 | if(open_file) { |
1839 | struct cifs_sb_info *cifs_sb; | ||
1840 | |||
1830 | /* there is not actually a write pending so let | 1841 | /* there is not actually a write pending so let |
1831 | this handle go free and allow it to | 1842 | this handle go free and allow it to |
1832 | be closable if needed */ | 1843 | be closable if needed */ |
1833 | atomic_dec(&open_file->wrtPending); | 1844 | atomic_dec(&open_file->wrtPending); |
1845 | |||
1846 | cifs_sb = CIFS_SB(cifsInode->vfs_inode.i_sb); | ||
1847 | if ( cifs_sb->mnt_cifs_flags & CIFS_MOUNT_DIRECT_IO ) { | ||
1848 | /* since no page cache to corrupt on directio | ||
1849 | we can change size safely */ | ||
1850 | return 1; | ||
1851 | } | ||
1852 | |||
1834 | return 0; | 1853 | return 0; |
1835 | } else | 1854 | } else |
1836 | return 1; | 1855 | return 1; |
@@ -1875,9 +1894,7 @@ struct address_space_operations cifs_addr_ops = { | |||
1875 | .readpage = cifs_readpage, | 1894 | .readpage = cifs_readpage, |
1876 | .readpages = cifs_readpages, | 1895 | .readpages = cifs_readpages, |
1877 | .writepage = cifs_writepage, | 1896 | .writepage = cifs_writepage, |
1878 | #ifdef CONFIG_CIFS_EXPERIMENTAL | ||
1879 | .writepages = cifs_writepages, | 1897 | .writepages = cifs_writepages, |
1880 | #endif | ||
1881 | .prepare_write = cifs_prepare_write, | 1898 | .prepare_write = cifs_prepare_write, |
1882 | .commit_write = cifs_commit_write, | 1899 | .commit_write = cifs_commit_write, |
1883 | .set_page_dirty = __set_page_dirty_nobuffers, | 1900 | .set_page_dirty = __set_page_dirty_nobuffers, |