|
@@ -675,6 +675,39 @@ void mark_buffer_dirty_inode(struct buffer_head *bh, struct inode *inode)
|
|
|
}
|
|
|
EXPORT_SYMBOL(mark_buffer_dirty_inode);
|
|
|
|
|
|
+/*
|
|
|
+ * Mark the page dirty, and set it dirty in the radix tree, and mark the inode
|
|
|
+ * dirty.
|
|
|
+ *
|
|
|
+ * If warn is true, then emit a warning if the page is not uptodate and has
|
|
|
+ * not been truncated.
|
|
|
+ */
|
|
|
+static int __set_page_dirty(struct page *page,
|
|
|
+ struct address_space *mapping, int warn)
|
|
|
+{
|
|
|
+ if (unlikely(!mapping))
|
|
|
+ return !TestSetPageDirty(page);
|
|
|
+
|
|
|
+ if (TestSetPageDirty(page))
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ write_lock_irq(&mapping->tree_lock);
|
|
|
+ if (page->mapping) { /* Race with truncate? */
|
|
|
+ WARN_ON_ONCE(warn && !PageUptodate(page));
|
|
|
+
|
|
|
+ if (mapping_cap_account_dirty(mapping)) {
|
|
|
+ __inc_zone_page_state(page, NR_FILE_DIRTY);
|
|
|
+ task_io_account_write(PAGE_CACHE_SIZE);
|
|
|
+ }
|
|
|
+ radix_tree_tag_set(&mapping->page_tree,
|
|
|
+ page_index(page), PAGECACHE_TAG_DIRTY);
|
|
|
+ }
|
|
|
+ write_unlock_irq(&mapping->tree_lock);
|
|
|
+ __mark_inode_dirty(mapping->host, I_DIRTY_PAGES);
|
|
|
+
|
|
|
+ return 1;
|
|
|
+}
|
|
|
+
|
|
|
/*
|
|
|
* Add a page to the dirty page list.
|
|
|
*
|
|
@@ -702,7 +735,7 @@ EXPORT_SYMBOL(mark_buffer_dirty_inode);
|
|
|
*/
|
|
|
int __set_page_dirty_buffers(struct page *page)
|
|
|
{
|
|
|
- struct address_space * const mapping = page_mapping(page);
|
|
|
+ struct address_space *mapping = page_mapping(page);
|
|
|
|
|
|
if (unlikely(!mapping))
|
|
|
return !TestSetPageDirty(page);
|
|
@@ -719,21 +752,7 @@ int __set_page_dirty_buffers(struct page *page)
|
|
|
}
|
|
|
spin_unlock(&mapping->private_lock);
|
|
|
|
|
|
- if (TestSetPageDirty(page))
|
|
|
- return 0;
|
|
|
-
|
|
|
- write_lock_irq(&mapping->tree_lock);
|
|
|
- if (page->mapping) { /* Race with truncate? */
|
|
|
- if (mapping_cap_account_dirty(mapping)) {
|
|
|
- __inc_zone_page_state(page, NR_FILE_DIRTY);
|
|
|
- task_io_account_write(PAGE_CACHE_SIZE);
|
|
|
- }
|
|
|
- radix_tree_tag_set(&mapping->page_tree,
|
|
|
- page_index(page), PAGECACHE_TAG_DIRTY);
|
|
|
- }
|
|
|
- write_unlock_irq(&mapping->tree_lock);
|
|
|
- __mark_inode_dirty(mapping->host, I_DIRTY_PAGES);
|
|
|
- return 1;
|
|
|
+ return __set_page_dirty(page, mapping, 1);
|
|
|
}
|
|
|
EXPORT_SYMBOL(__set_page_dirty_buffers);
|
|
|
|
|
@@ -1132,8 +1151,9 @@ __getblk_slow(struct block_device *bdev, sector_t block, int size)
|
|
|
*/
|
|
|
void fastcall mark_buffer_dirty(struct buffer_head *bh)
|
|
|
{
|
|
|
+ WARN_ON_ONCE(!buffer_uptodate(bh));
|
|
|
if (!buffer_dirty(bh) && !test_set_buffer_dirty(bh))
|
|
|
- __set_page_dirty_nobuffers(bh->b_page);
|
|
|
+ __set_page_dirty(bh->b_page, page_mapping(bh->b_page), 0);
|
|
|
}
|
|
|
|
|
|
/*
|