diff --git a/fs/reiser4/as_ops.c b/fs/reiser4/as_ops.c index decb9eb21577d09c18cb877c08cd1c9a04ad36a2..68b9e8127a7596f289cc8469292b0c132b61fefd 100644 --- a/fs/reiser4/as_ops.c +++ b/fs/reiser4/as_ops.c @@ -347,24 +347,6 @@ int reiser4_writepages(struct address_space *mapping, return inode_file_plugin(mapping->host)->writepages(mapping, wbc); } -int reiser4_prepare_write(struct file *file, struct page *page, - unsigned from, unsigned to) -{ - return inode_file_plugin(file->f_dentry->d_inode)->prepare_write(file, - page, - from, - to); -} - -int reiser4_commit_write(struct file *file, struct page *page, - unsigned from, unsigned to) -{ - return inode_file_plugin(file->f_dentry->d_inode)->commit_write(file, - page, - from, - to); -} - /* Make Linus happy. Local variables: c-indentation-style: "K&R" diff --git a/fs/reiser4/page_cache.c b/fs/reiser4/page_cache.c index 654e7ae7313857055cc94d54960641d9b9f9b92c..6c69285a7aa832f52c1ac5b30957f37aeb72c860 100644 --- a/fs/reiser4/page_cache.c +++ b/fs/reiser4/page_cache.c @@ -560,8 +560,8 @@ static struct address_space_operations formatted_fake_as_ops = { .set_page_dirty = formatted_set_page_dirty, /* used for read-ahead. Not applicable */ .readpages = NULL, - .prepare_write = NULL, - .commit_write = NULL, + .write_begin = NULL, + .write_end = NULL, .bmap = NULL, /* called just before page is being detached from inode mapping and removed from memory. Called on truncate, cut/squeeze, and diff --git a/fs/reiser4/plugin/file/cryptcompress.c b/fs/reiser4/plugin/file/cryptcompress.c index 7d3061de6ac623ada6d17d4682a653a3bf0706ce..b196209c1499f5100c5a2dd97389b83d1e61e095 100644 --- a/fs/reiser4/plugin/file/cryptcompress.c +++ b/fs/reiser4/plugin/file/cryptcompress.c @@ -3405,11 +3405,12 @@ static int cryptcompress_truncate(struct inode *inode, /* old size */ return result; } -/* Capture an anonymous pager cluster. (Page cluser is - * anonymous if it contains at least one anonymous page +/** + * Capture a pager cluster. + * @clust must be set up by a caller. */ -static int capture_anon_page_cluster(struct cluster_handle * clust, - struct inode * inode) +static int capture_page_cluster(struct cluster_handle * clust, + struct inode * inode) { int result; @@ -3420,6 +3421,7 @@ static int capture_anon_page_cluster(struct cluster_handle * clust, result = prepare_logical_cluster(inode, 0, 0, clust, LC_APPOV); if (result) return result; + set_cluster_pages_dirty(clust, inode); result = checkin_logical_cluster(clust, inode); put_hint_cluster(clust, inode, ZNODE_WRITE_LOCK); @@ -3502,7 +3504,7 @@ static int capture_anon_pages(struct address_space * mapping, pgoff_t * index, break; } move_cluster_forward(&clust, inode, pages[0]->index); - result = capture_anon_page_cluster(&clust, inode); + result = capture_page_cluster(&clust, inode); put_found_pages(pages, found); /* find_anon_page_cluster */ if (result) @@ -3743,18 +3745,48 @@ int release_cryptcompress(struct inode *inode, struct file *file) } /* plugin->prepare_write */ -int prepare_write_cryptcompress(struct file *file, struct page *page, - unsigned from, unsigned to) +int write_begin_cryptcompress(struct file *file, struct page *page, + unsigned from, unsigned to) { - return -EINVAL; + return do_prepare_write(file, page, from, to); } /* plugin->commit_write */ -int commit_write_cryptcompress(struct file *file, struct page *page, - unsigned from, unsigned to) +int write_end_cryptcompress(struct file *file, struct page *page, + unsigned from, unsigned to) { - BUG(); - return 0; + int ret; + hint_t *hint; + lock_handle *lh; + struct inode * inode; + struct cluster_handle clust; + + unlock_page(page); + + inode = page->mapping->host; + hint = kmalloc(sizeof(*hint), reiser4_ctx_gfp_mask_get()); + if (hint == NULL) + return RETERR(-ENOMEM); + hint_init_zero(hint); + lh = &hint->lh; + + cluster_init_read(&clust, NULL); + clust.hint = hint; + + ret = alloc_cluster_pgset(&clust, cluster_nrpages(inode)); + if (ret) + goto out; + clust.index = pg_to_clust(page->index, inode); + ret = capture_page_cluster(&clust, inode); + if (ret) + warning("edward-1557", + "Capture failed (inode %llu, result=%i)", + (unsigned long long)get_inode_oid(inode), ret); + out: + done_lh(lh); + kfree(hint); + put_cluster_handle(&clust); + return ret; } /* plugin->bmap */ diff --git a/fs/reiser4/plugin/file/file.c b/fs/reiser4/plugin/file/file.c index 783e99fc4df0980668ebdb2c7d55a12f88f7f9aa..1aee5ad310cb4c1bba1277ed14c6843612ad0232 100644 --- a/fs/reiser4/plugin/file/file.c +++ b/fs/reiser4/plugin/file/file.c @@ -889,36 +889,12 @@ static int capture_page_and_create_extent(struct page *page) return result; } -/* this is implementation of method commit_write of struct - address_space_operations for unix file plugin */ -int -commit_write_unix_file(struct file *file, struct page *page, - unsigned from, unsigned to) +/* plugin->write_end() */ +int write_end_unix_file(struct file *file, struct page *page, + unsigned from, unsigned to) { - reiser4_context *ctx; - struct inode *inode; - int result; - - assert("umka-3101", file != NULL); - assert("umka-3102", page != NULL); - assert("umka-3093", PageLocked(page)); - - SetPageUptodate(page); - - inode = page->mapping->host; - ctx = reiser4_init_context(page->mapping->host->i_sb); - if (IS_ERR(ctx)) - return PTR_ERR(ctx); - page_cache_get(page); unlock_page(page); - result = capture_page_and_create_extent(page); - lock_page(page); - page_cache_release(page); - - /* don't commit transaction under inode semaphore */ - context_set_commit_async(ctx); - reiser4_exit_context(ctx); - return result; + return capture_page_and_create_extent(page); } /* @@ -2687,32 +2663,23 @@ int delete_object_unix_file(struct inode *inode) return reiser4_delete_object_common(inode); } -int -prepare_write_unix_file(struct file *file, struct page *page, - unsigned from, unsigned to) +/* plugin->write_begin() */ +int write_begin_unix_file(struct file *file, struct page *page, + unsigned from, unsigned to) { - reiser4_context *ctx; - struct unix_file_info *uf_info; int ret; + struct unix_file_info *info; - ctx = reiser4_init_context(file->f_dentry->d_inode->i_sb); - if (IS_ERR(ctx)) - return PTR_ERR(ctx); - - uf_info = unix_file_inode_data(file->f_dentry->d_inode); - get_exclusive_access(uf_info); - ret = find_file_state(file->f_dentry->d_inode, uf_info); - if (ret == 0) { - if (uf_info->container == UF_CONTAINER_TAILS) + info = unix_file_inode_data(file->f_dentry->d_inode); + get_exclusive_access(info); + ret = find_file_state(file->f_dentry->d_inode, info); + if (likely(ret == 0)) { + if (info->container == UF_CONTAINER_TAILS) ret = -EINVAL; else ret = do_prepare_write(file, page, from, to); } - drop_exclusive_access(uf_info); - - /* don't commit transaction under inode semaphore */ - context_set_commit_async(ctx); - reiser4_exit_context(ctx); + drop_exclusive_access(info); return ret; } diff --git a/fs/reiser4/plugin/file/file.h b/fs/reiser4/plugin/file/file.h index 656c8b094bfcf29b89f9afbac220e01a391da080..f74498cd17c23f02f6d6cd97c23bf6f74b41f851 100644 --- a/fs/reiser4/plugin/file/file.h +++ b/fs/reiser4/plugin/file/file.h @@ -59,10 +59,14 @@ int reiser4_readpage(struct file *, struct page *); int reiser4_readpages(struct file*, struct address_space*, struct list_head*, unsigned); int reiser4_writepages(struct address_space *, struct writeback_control *); -int reiser4_prepare_write(struct file *, struct page *, unsigned from, - unsigned to); -int reiser4_commit_write(struct file *, struct page *, unsigned from, - unsigned to); +int reiser4_write_begin_careful(struct file *file, + struct address_space *mapping, + loff_t pos, unsigned len, unsigned flags, + struct page **pagep, void **fsdata); +int reiser4_write_end_careful(struct file *file, + struct address_space *mapping, + loff_t pos, unsigned len, unsigned copied, + struct page *page, void *fsdata); sector_t reiser4_bmap_careful(struct address_space *, sector_t lblock); /* @@ -87,12 +91,13 @@ int release_unix_file(struct inode *, struct file *); /* private address space operations */ int readpage_unix_file(struct file *, struct page *); -int readpages_unix_file(struct file*, struct address_space*, struct list_head*, unsigned); +int readpages_unix_file(struct file*, struct address_space*, struct list_head*, + unsigned); int writepages_unix_file(struct address_space *, struct writeback_control *); -int prepare_write_unix_file(struct file *, struct page *, unsigned from, - unsigned to); -int commit_write_unix_file(struct file *, struct page *, unsigned from, - unsigned to); +int write_begin_unix_file(struct file *file, struct page *page, + unsigned from, unsigned to); +int write_end_unix_file(struct file *file, struct page *page, + unsigned from, unsigned to); sector_t bmap_unix_file(struct address_space *, sector_t lblock); /* other private methods */ @@ -129,10 +134,10 @@ int readpages_cryptcompress(struct file*, struct address_space*, struct list_head*, unsigned); int writepages_cryptcompress(struct address_space *, struct writeback_control *); -int prepare_write_cryptcompress(struct file *, struct page *, unsigned from, - unsigned to); -int commit_write_cryptcompress(struct file *, struct page *, unsigned from, - unsigned to); +int write_begin_cryptcompress(struct file *file, struct page *page, + unsigned from, unsigned to); +int write_end_cryptcompress(struct file *file, struct page *page, + unsigned from, unsigned to); sector_t bmap_cryptcompress(struct address_space *, sector_t lblock); /* other private methods */ diff --git a/fs/reiser4/plugin/file/file_conversion.c b/fs/reiser4/plugin/file/file_conversion.c index 1f0648283af69fbe07d400947b005a3987c64343..7392876767483cf80e4dd55bcd15545777b3887e 100644 --- a/fs/reiser4/plugin/file/file_conversion.c +++ b/fs/reiser4/plugin/file/file_conversion.c @@ -667,6 +667,89 @@ sector_t reiser4_bmap_careful(struct address_space * mapping, sector_t lblock) return PROT_PASSIVE(sector_t, bmap, (mapping, lblock)); } +int reiser4_write_begin_careful(struct file *file, + struct address_space *mapping, + loff_t pos, + unsigned len, + unsigned flags, + struct page **pagep, + void **fsdata) +{ + int ret = 0; + unsigned start, end; + struct page *page; + pgoff_t index; + reiser4_context *ctx; + struct inode * inode = file->f_dentry->d_inode; + + index = pos >> PAGE_CACHE_SHIFT; + start = pos & (PAGE_CACHE_SIZE - 1); + end = start + len; + + page = __grab_cache_page(mapping, index); + *pagep = page; + if (!page) + return -ENOMEM; + + ctx = reiser4_init_context(file->f_dentry->d_inode->i_sb); + if (IS_ERR(ctx)) { + ret = PTR_ERR(ctx); + goto out; + } + ret = PROT_PASSIVE(int, write_begin, (file, page, start, end)); + + /* don't commit transaction under inode semaphore */ + context_set_commit_async(ctx); + reiser4_exit_context(ctx); + out: + if (unlikely(ret)) { + unlock_page(page); + page_cache_release(page); + } + return ret; +} + +int reiser4_write_end_careful(struct file *file, + struct address_space *mapping, + loff_t pos, + unsigned len, + unsigned copied, + struct page *page, + void *fsdata) +{ + int ret; + reiser4_context *ctx; + unsigned start, end; + struct inode *inode = page->mapping->host; + + assert("umka-3101", file != NULL); + assert("umka-3102", page != NULL); + assert("umka-3093", PageLocked(page)); + + start = pos & (PAGE_CACHE_SIZE - 1); + end = start + len; + + flush_dcache_page(page); + SetPageUptodate(page); + + ctx = reiser4_init_context(page->mapping->host->i_sb); + if (IS_ERR(ctx)){ + unlock_page(page); + ret = PTR_ERR(ctx); + goto out; + } + ret = PROT_PASSIVE(int, write_end, (file, page, start, end)); + + /* don't commit transaction under inode semaphore */ + context_set_commit_async(ctx); + reiser4_exit_context(ctx); + out: + page_cache_release(page); + if (!ret) + ret = copied; + return ret; +} + /* * Wrappers without protection for: * diff --git a/fs/reiser4/plugin/file_ops.c b/fs/reiser4/plugin/file_ops.c index d4fb42c3ba5e2e42c70bbcfcf325d83dee68de03..135f8a751315273304ab9c1015cabb81336c00b3 100644 --- a/fs/reiser4/plugin/file_ops.c +++ b/fs/reiser4/plugin/file_ops.c @@ -95,55 +95,12 @@ int reiser4_sync_file_common(struct file *file, return 0; } -/* this is common implementation of vfs's sendfile method of struct - file_operations - - Reads @count bytes from @file and calls @actor for every page read. This is - needed for loop back devices support. -*/ -#if 0 -ssize_t -sendfile_common(struct file *file, loff_t *ppos, size_t count, - read_actor_t actor, void *target) -{ - reiser4_context *ctx; - ssize_t result; - - ctx = reiser4_init_context(file->f_dentry->d_inode->i_sb); - if (IS_ERR(ctx)) - return PTR_ERR(ctx); - result = generic_file_sendfile(file, ppos, count, actor, target); - reiser4_exit_context(ctx); - return result; -} -#endif /* 0 */ /* address space operations */ -/* this is common implementation of vfs's prepare_write method of struct - address_space_operations -*/ -int -prepare_write_common(struct file *file, struct page *page, unsigned from, - unsigned to) -{ - reiser4_context *ctx; - int result; - ctx = reiser4_init_context(page->mapping->host->i_sb); - result = do_prepare_write(file, page, from, to); - - /* don't commit transaction under inode semaphore */ - context_set_commit_async(ctx); - reiser4_exit_context(ctx); - - return result; -} - -/* this is helper for prepare_write_common and prepare_write_unix_file - */ -int -do_prepare_write(struct file *file, struct page *page, unsigned from, +/* this is helper for plugin->write_begin() */ +int do_prepare_write(struct file *file, struct page *page, unsigned from, unsigned to) { int result; diff --git a/fs/reiser4/plugin/object.c b/fs/reiser4/plugin/object.c index a2178aa384d280ad54ffef2e02a0705f6be3d3df..722f0a261fc513cee991b364482ff7ed6ece4521 100644 --- a/fs/reiser4/plugin/object.c +++ b/fs/reiser4/plugin/object.c @@ -114,8 +114,8 @@ static struct address_space_operations regular_file_a_ops = { .writepages = reiser4_writepages, .set_page_dirty = reiser4_set_page_dirty, .readpages = reiser4_readpages, - .prepare_write = reiser4_prepare_write, - .commit_write = reiser4_commit_write, + .write_begin = reiser4_write_begin_careful, + .write_end = reiser4_write_end_careful, .bmap = reiser4_bmap_careful, .invalidatepage = reiser4_invalidatepage, .releasepage = reiser4_releasepage @@ -165,8 +165,8 @@ static struct address_space_operations directory_a_ops = { .writepages = dummyop, .set_page_dirty = bugop, .readpages = bugop, - .prepare_write = bugop, - .commit_write = bugop, + .write_begin = bugop, + .write_end = bugop, .bmap = bugop, .invalidatepage = bugop, .releasepage = bugop @@ -209,8 +209,8 @@ file_plugin file_plugins[LAST_FILE_PLUGIN_ID] = { .readpage = readpage_unix_file, .readpages = readpages_unix_file, .writepages = writepages_unix_file, - .prepare_write = prepare_write_unix_file, - .commit_write = commit_write_unix_file, + .write_begin = write_begin_unix_file, + .write_end = write_end_unix_file, /* * private a_ops */ @@ -403,8 +403,8 @@ file_plugin file_plugins[LAST_FILE_PLUGIN_ID] = { .readpage = readpage_cryptcompress, .readpages = readpages_cryptcompress, .writepages = writepages_cryptcompress, - .prepare_write = prepare_write_cryptcompress, - .commit_write = commit_write_cryptcompress, + .write_begin = write_begin_cryptcompress, + .write_end = write_end_cryptcompress, .bmap = bmap_cryptcompress, diff --git a/fs/reiser4/plugin/object.h b/fs/reiser4/plugin/object.h index 88a53a58b084312771b4c635f9a67bcb2c6a724c..ed580cadb3ef5d453b1c8097d50fa0b9ed6c0d3c 100644 --- a/fs/reiser4/plugin/object.h +++ b/fs/reiser4/plugin/object.h @@ -36,9 +36,6 @@ int reiser4_readdir_common(struct file *, void *dirent, filldir_t); int reiser4_release_dir_common(struct inode *, struct file *); int reiser4_sync_common(struct file *, struct dentry *, int datasync); -/* common implementations of address space operations */ -int prepare_write_common(struct file *, struct page *, unsigned from, - unsigned to); /* file plugin operations: common implementations */ int write_sd_by_inode_common(struct inode *); diff --git a/fs/reiser4/plugin/plugin.h b/fs/reiser4/plugin/plugin.h index b2221b57d98fa5a9007e705be275faae69dc6dc3..76ef56c6249d2266dd6c71afe239823d60c24141 100644 --- a/fs/reiser4/plugin/plugin.h +++ b/fs/reiser4/plugin/plugin.h @@ -248,10 +248,10 @@ typedef struct file_plugin { struct list_head *pages, unsigned nr_pages); int (*writepages)(struct address_space *mapping, struct writeback_control *wbc); - int (*prepare_write)(struct file *file, struct page *page, - unsigned from, unsigned to); - int (*commit_write)(struct file *file, struct page *page, - unsigned from, unsigned to); + int (*write_begin)(struct file *file, struct page *page, + unsigned from, unsigned to); + int (*write_end)(struct file *file, struct page *page, + unsigned from, unsigned to); sector_t (*bmap) (struct address_space * mapping, sector_t lblock); /* other private methods */ /* save inode cached stat-data onto disk. It was called