Commit 3fdcfb66 authored by Tao Ma's avatar Tao Ma Committed by Theodore Ts'o

ext4: add journalled write support for inline data

Signed-off-by: default avatarTao Ma <boyu.mt@taobao.com>
Signed-off-by: default avatar"Theodore Ts'o" <tytso@mit.edu>
parent f19d5870
...@@ -747,6 +747,30 @@ int ext4_write_inline_data_end(struct inode *inode, loff_t pos, unsigned len, ...@@ -747,6 +747,30 @@ int ext4_write_inline_data_end(struct inode *inode, loff_t pos, unsigned len,
return copied; return copied;
} }
struct buffer_head *
ext4_journalled_write_inline_data(struct inode *inode,
unsigned len,
struct page *page)
{
int ret;
void *kaddr;
struct ext4_iloc iloc;
ret = ext4_get_inode_loc(inode, &iloc);
if (ret) {
ext4_std_error(inode->i_sb, ret);
return NULL;
}
down_write(&EXT4_I(inode)->xattr_sem);
kaddr = kmap_atomic(page);
ext4_write_inline_data(inode, &iloc, kaddr, 0, len);
kunmap_atomic(kaddr);
up_write(&EXT4_I(inode)->xattr_sem);
return iloc.bh;
}
int ext4_destroy_inline_data(handle_t *handle, struct inode *inode) int ext4_destroy_inline_data(handle_t *handle, struct inode *inode)
{ {
......
...@@ -1124,6 +1124,10 @@ static int ext4_journalled_write_end(struct file *file, ...@@ -1124,6 +1124,10 @@ static int ext4_journalled_write_end(struct file *file,
BUG_ON(!ext4_handle_valid(handle)); BUG_ON(!ext4_handle_valid(handle));
if (ext4_has_inline_data(inode))
copied = ext4_write_inline_data_end(inode, pos, len,
copied, page);
else {
if (copied < len) { if (copied < len) {
if (!PageUptodate(page)) if (!PageUptodate(page))
copied = 0; copied = 0;
...@@ -1134,6 +1138,7 @@ static int ext4_journalled_write_end(struct file *file, ...@@ -1134,6 +1138,7 @@ static int ext4_journalled_write_end(struct file *file,
to, &partial, write_end_fn); to, &partial, write_end_fn);
if (!partial) if (!partial)
SetPageUptodate(page); SetPageUptodate(page);
}
new_i_size = pos + copied; new_i_size = pos + copied;
if (new_i_size > inode->i_size) if (new_i_size > inode->i_size)
i_size_write(inode, pos+copied); i_size_write(inode, pos+copied);
...@@ -1911,15 +1916,29 @@ static int __ext4_journalled_writepage(struct page *page, ...@@ -1911,15 +1916,29 @@ static int __ext4_journalled_writepage(struct page *page,
{ {
struct address_space *mapping = page->mapping; struct address_space *mapping = page->mapping;
struct inode *inode = mapping->host; struct inode *inode = mapping->host;
struct buffer_head *page_bufs; struct buffer_head *page_bufs = NULL;
handle_t *handle = NULL; handle_t *handle = NULL;
int ret = 0; int ret = 0, err = 0;
int err; int inline_data = ext4_has_inline_data(inode);
struct buffer_head *inode_bh = NULL;
ClearPageChecked(page); ClearPageChecked(page);
if (inline_data) {
BUG_ON(page->index != 0);
BUG_ON(len > ext4_get_max_inline_size(inode));
inode_bh = ext4_journalled_write_inline_data(inode, len, page);
if (inode_bh == NULL)
goto out;
} else {
page_bufs = page_buffers(page); page_bufs = page_buffers(page);
BUG_ON(!page_bufs); if (!page_bufs) {
ext4_walk_page_buffers(handle, page_bufs, 0, len, NULL, bget_one); BUG();
goto out;
}
ext4_walk_page_buffers(handle, page_bufs, 0, len,
NULL, bget_one);
}
/* As soon as we unlock the page, it can go away, but we have /* As soon as we unlock the page, it can go away, but we have
* references to buffers so we are safe */ * references to buffers so we are safe */
unlock_page(page); unlock_page(page);
...@@ -1932,11 +1951,18 @@ static int __ext4_journalled_writepage(struct page *page, ...@@ -1932,11 +1951,18 @@ static int __ext4_journalled_writepage(struct page *page,
BUG_ON(!ext4_handle_valid(handle)); BUG_ON(!ext4_handle_valid(handle));
if (inline_data) {
ret = ext4_journal_get_write_access(handle, inode_bh);
err = ext4_handle_dirty_metadata(handle, inode, inode_bh);
} else {
ret = ext4_walk_page_buffers(handle, page_bufs, 0, len, NULL, ret = ext4_walk_page_buffers(handle, page_bufs, 0, len, NULL,
do_journal_get_write_access); do_journal_get_write_access);
err = ext4_walk_page_buffers(handle, page_bufs, 0, len, NULL, err = ext4_walk_page_buffers(handle, page_bufs, 0, len, NULL,
write_end_fn); write_end_fn);
}
if (ret == 0) if (ret == 0)
ret = err; ret = err;
EXT4_I(inode)->i_datasync_tid = handle->h_transaction->t_tid; EXT4_I(inode)->i_datasync_tid = handle->h_transaction->t_tid;
...@@ -1944,9 +1970,12 @@ static int __ext4_journalled_writepage(struct page *page, ...@@ -1944,9 +1970,12 @@ static int __ext4_journalled_writepage(struct page *page,
if (!ret) if (!ret)
ret = err; ret = err;
ext4_walk_page_buffers(handle, page_bufs, 0, len, NULL, bput_one); if (!ext4_has_inline_data(inode))
ext4_walk_page_buffers(handle, page_bufs, 0, len,
NULL, bput_one);
ext4_set_inode_state(inode, EXT4_STATE_JDATA); ext4_set_inode_state(inode, EXT4_STATE_JDATA);
out: out:
brelse(inode_bh);
return ret; return ret;
} }
......
...@@ -150,6 +150,10 @@ extern int ext4_write_inline_data_end(struct inode *inode, ...@@ -150,6 +150,10 @@ extern int ext4_write_inline_data_end(struct inode *inode,
loff_t pos, unsigned len, loff_t pos, unsigned len,
unsigned copied, unsigned copied,
struct page *page); struct page *page);
extern struct buffer_head *
ext4_journalled_write_inline_data(struct inode *inode,
unsigned len,
struct page *page);
# else /* CONFIG_EXT4_FS_XATTR */ # else /* CONFIG_EXT4_FS_XATTR */
static inline int static inline int
...@@ -288,6 +292,14 @@ static inline int ext4_write_inline_data_end(struct inode *inode, ...@@ -288,6 +292,14 @@ static inline int ext4_write_inline_data_end(struct inode *inode,
{ {
return 0; return 0;
} }
static inline struct buffer_head *
ext4_journalled_write_inline_data(struct inode *inode,
unsigned len,
struct page *page)
{
return NULL;
}
# endif /* CONFIG_EXT4_FS_XATTR */ # endif /* CONFIG_EXT4_FS_XATTR */
#ifdef CONFIG_EXT4_FS_SECURITY #ifdef CONFIG_EXT4_FS_SECURITY
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment