diff options
Diffstat (limited to 'mm/filemap.c')
-rw-r--r-- | mm/filemap.c | 31 |
1 files changed, 29 insertions, 2 deletions
diff --git a/mm/filemap.c b/mm/filemap.c index a4a5260b0279..fa5ca304148e 100644 --- a/mm/filemap.c +++ b/mm/filemap.c | |||
@@ -1712,8 +1712,35 @@ page_not_uptodate: | |||
1712 | } | 1712 | } |
1713 | EXPORT_SYMBOL(filemap_fault); | 1713 | EXPORT_SYMBOL(filemap_fault); |
1714 | 1714 | ||
1715 | int filemap_page_mkwrite(struct vm_area_struct *vma, struct vm_fault *vmf) | ||
1716 | { | ||
1717 | struct page *page = vmf->page; | ||
1718 | struct inode *inode = vma->vm_file->f_path.dentry->d_inode; | ||
1719 | int ret = VM_FAULT_LOCKED; | ||
1720 | |||
1721 | sb_start_pagefault(inode->i_sb); | ||
1722 | file_update_time(vma->vm_file); | ||
1723 | lock_page(page); | ||
1724 | if (page->mapping != inode->i_mapping) { | ||
1725 | unlock_page(page); | ||
1726 | ret = VM_FAULT_NOPAGE; | ||
1727 | goto out; | ||
1728 | } | ||
1729 | /* | ||
1730 | * We mark the page dirty already here so that when freeze is in | ||
1731 | * progress, we are guaranteed that writeback during freezing will | ||
1732 | * see the dirty page and writeprotect it again. | ||
1733 | */ | ||
1734 | set_page_dirty(page); | ||
1735 | out: | ||
1736 | sb_end_pagefault(inode->i_sb); | ||
1737 | return ret; | ||
1738 | } | ||
1739 | EXPORT_SYMBOL(filemap_page_mkwrite); | ||
1740 | |||
1715 | const struct vm_operations_struct generic_file_vm_ops = { | 1741 | const struct vm_operations_struct generic_file_vm_ops = { |
1716 | .fault = filemap_fault, | 1742 | .fault = filemap_fault, |
1743 | .page_mkwrite = filemap_page_mkwrite, | ||
1717 | }; | 1744 | }; |
1718 | 1745 | ||
1719 | /* This is used for a general mmap of a disk file */ | 1746 | /* This is used for a general mmap of a disk file */ |
@@ -2407,8 +2434,6 @@ ssize_t __generic_file_aio_write(struct kiocb *iocb, const struct iovec *iov, | |||
2407 | count = ocount; | 2434 | count = ocount; |
2408 | pos = *ppos; | 2435 | pos = *ppos; |
2409 | 2436 | ||
2410 | vfs_check_frozen(inode->i_sb, SB_FREEZE_WRITE); | ||
2411 | |||
2412 | /* We can write back this queue in page reclaim */ | 2437 | /* We can write back this queue in page reclaim */ |
2413 | current->backing_dev_info = mapping->backing_dev_info; | 2438 | current->backing_dev_info = mapping->backing_dev_info; |
2414 | written = 0; | 2439 | written = 0; |
@@ -2507,6 +2532,7 @@ ssize_t generic_file_aio_write(struct kiocb *iocb, const struct iovec *iov, | |||
2507 | 2532 | ||
2508 | BUG_ON(iocb->ki_pos != pos); | 2533 | BUG_ON(iocb->ki_pos != pos); |
2509 | 2534 | ||
2535 | sb_start_write(inode->i_sb); | ||
2510 | mutex_lock(&inode->i_mutex); | 2536 | mutex_lock(&inode->i_mutex); |
2511 | blk_start_plug(&plug); | 2537 | blk_start_plug(&plug); |
2512 | ret = __generic_file_aio_write(iocb, iov, nr_segs, &iocb->ki_pos); | 2538 | ret = __generic_file_aio_write(iocb, iov, nr_segs, &iocb->ki_pos); |
@@ -2520,6 +2546,7 @@ ssize_t generic_file_aio_write(struct kiocb *iocb, const struct iovec *iov, | |||
2520 | ret = err; | 2546 | ret = err; |
2521 | } | 2547 | } |
2522 | blk_finish_plug(&plug); | 2548 | blk_finish_plug(&plug); |
2549 | sb_end_write(inode->i_sb); | ||
2523 | return ret; | 2550 | return ret; |
2524 | } | 2551 | } |
2525 | EXPORT_SYMBOL(generic_file_aio_write); | 2552 | EXPORT_SYMBOL(generic_file_aio_write); |