ext4: add journalled write support for inline data
Signed-off-by: Tao Ma <boyu.mt@taobao.com> Signed-off-by: "Theodore Ts'o" <tytso@mit.edu>
This commit is contained in:
@@ -747,6 +747,30 @@ out:
|
|||||||
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,16 +1124,21 @@ static int ext4_journalled_write_end(struct file *file,
|
|||||||
|
|
||||||
BUG_ON(!ext4_handle_valid(handle));
|
BUG_ON(!ext4_handle_valid(handle));
|
||||||
|
|
||||||
if (copied < len) {
|
if (ext4_has_inline_data(inode))
|
||||||
if (!PageUptodate(page))
|
copied = ext4_write_inline_data_end(inode, pos, len,
|
||||||
copied = 0;
|
copied, page);
|
||||||
page_zero_new_buffers(page, from+copied, to);
|
else {
|
||||||
}
|
if (copied < len) {
|
||||||
|
if (!PageUptodate(page))
|
||||||
|
copied = 0;
|
||||||
|
page_zero_new_buffers(page, from+copied, to);
|
||||||
|
}
|
||||||
|
|
||||||
ret = ext4_walk_page_buffers(handle, page_buffers(page), from,
|
ret = ext4_walk_page_buffers(handle, page_buffers(page), from,
|
||||||
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,
|
|||||||
{
|
{
|
||||||
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);
|
||||||
page_bufs = page_buffers(page);
|
|
||||||
BUG_ON(!page_bufs);
|
if (inline_data) {
|
||||||
ext4_walk_page_buffers(handle, page_bufs, 0, len, NULL, bget_one);
|
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);
|
||||||
|
if (!page_bufs) {
|
||||||
|
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,
|
|||||||
|
|
||||||
BUG_ON(!ext4_handle_valid(handle));
|
BUG_ON(!ext4_handle_valid(handle));
|
||||||
|
|
||||||
ret = ext4_walk_page_buffers(handle, page_bufs, 0, len, NULL,
|
if (inline_data) {
|
||||||
do_journal_get_write_access);
|
ret = ext4_journal_get_write_access(handle, inode_bh);
|
||||||
|
|
||||||
err = ext4_walk_page_buffers(handle, page_bufs, 0, len, NULL,
|
err = ext4_handle_dirty_metadata(handle, inode, inode_bh);
|
||||||
write_end_fn);
|
|
||||||
|
} else {
|
||||||
|
ret = ext4_walk_page_buffers(handle, page_bufs, 0, len, NULL,
|
||||||
|
do_journal_get_write_access);
|
||||||
|
|
||||||
|
err = ext4_walk_page_buffers(handle, page_bufs, 0, len, NULL,
|
||||||
|
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,
|
|||||||
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,
|
|||||||
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,
|
|||||||
{
|
{
|
||||||
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
|
||||||
|
Reference in New Issue
Block a user