Commit 4645855d authored by Sasha Finkelstein's avatar Sasha Finkelstein Committed by Kent Overstreet

bcachefs: Hook up RENAME_WHITEOUT in rename.

This is needed for overlayfs, which is used by container managers.
Signed-off-by: default avatarSasha Finkelstein <fnkl.kernel@gmail.com>
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent d90c8acd
...@@ -529,6 +529,12 @@ void bch2_set_btree_iter_dontneed(struct btree_iter *); ...@@ -529,6 +529,12 @@ void bch2_set_btree_iter_dontneed(struct btree_iter *);
void *__bch2_trans_kmalloc(struct btree_trans *, size_t); void *__bch2_trans_kmalloc(struct btree_trans *, size_t);
/**
* bch2_trans_kmalloc - allocate memory for use by the current transaction
*
* Must be called after bch2_trans_begin, which on second and further calls
* frees all memory allocated in this transaction
*/
static inline void *bch2_trans_kmalloc(struct btree_trans *trans, size_t size) static inline void *bch2_trans_kmalloc(struct btree_trans *trans, size_t size)
{ {
size = roundup(size, 8); size = roundup(size, 8);
......
...@@ -42,7 +42,8 @@ int bch2_create_trans(struct btree_trans *trans, ...@@ -42,7 +42,8 @@ int bch2_create_trans(struct btree_trans *trans,
if (ret) if (ret)
goto err; goto err;
ret = bch2_inode_peek(trans, &dir_iter, dir_u, dir, BTREE_ITER_intent); ret = bch2_inode_peek(trans, &dir_iter, dir_u, dir,
BTREE_ITER_intent|BTREE_ITER_with_updates);
if (ret) if (ret)
goto err; goto err;
...@@ -163,7 +164,7 @@ int bch2_create_trans(struct btree_trans *trans, ...@@ -163,7 +164,7 @@ int bch2_create_trans(struct btree_trans *trans,
name, name,
dir_target, dir_target,
&dir_offset, &dir_offset,
STR_HASH_must_create); STR_HASH_must_create|BTREE_ITER_with_updates);
if (ret) if (ret)
goto err; goto err;
......
...@@ -736,15 +736,16 @@ static int bch2_rename2(struct mnt_idmap *idmap, ...@@ -736,15 +736,16 @@ static int bch2_rename2(struct mnt_idmap *idmap,
struct bch_inode_info *src_inode = to_bch_ei(src_dentry->d_inode); struct bch_inode_info *src_inode = to_bch_ei(src_dentry->d_inode);
struct bch_inode_info *dst_inode = to_bch_ei(dst_dentry->d_inode); struct bch_inode_info *dst_inode = to_bch_ei(dst_dentry->d_inode);
struct bch_inode_unpacked dst_dir_u, src_dir_u; struct bch_inode_unpacked dst_dir_u, src_dir_u;
struct bch_inode_unpacked src_inode_u, dst_inode_u; struct bch_inode_unpacked src_inode_u, dst_inode_u, *whiteout_inode_u;
struct btree_trans *trans; struct btree_trans *trans;
enum bch_rename_mode mode = flags & RENAME_EXCHANGE enum bch_rename_mode mode = flags & RENAME_EXCHANGE
? BCH_RENAME_EXCHANGE ? BCH_RENAME_EXCHANGE
: dst_dentry->d_inode : dst_dentry->d_inode
? BCH_RENAME_OVERWRITE : BCH_RENAME; ? BCH_RENAME_OVERWRITE : BCH_RENAME;
bool whiteout = !!(flags & RENAME_WHITEOUT);
int ret; int ret;
if (flags & ~(RENAME_NOREPLACE|RENAME_EXCHANGE)) if (flags & ~(RENAME_NOREPLACE|RENAME_EXCHANGE|RENAME_WHITEOUT))
return -EINVAL; return -EINVAL;
if (mode == BCH_RENAME_OVERWRITE) { if (mode == BCH_RENAME_OVERWRITE) {
...@@ -785,18 +786,48 @@ static int bch2_rename2(struct mnt_idmap *idmap, ...@@ -785,18 +786,48 @@ static int bch2_rename2(struct mnt_idmap *idmap,
if (ret) if (ret)
goto err; goto err;
} }
retry:
bch2_trans_begin(trans);
ret = commit_do(trans, NULL, NULL, 0, ret = bch2_rename_trans(trans,
bch2_rename_trans(trans, inode_inum(src_dir), &src_dir_u,
inode_inum(src_dir), &src_dir_u, inode_inum(dst_dir), &dst_dir_u,
inode_inum(dst_dir), &dst_dir_u, &src_inode_u,
&src_inode_u, &dst_inode_u,
&dst_inode_u, &src_dentry->d_name,
&src_dentry->d_name, &dst_dentry->d_name,
&dst_dentry->d_name, mode);
mode));
if (unlikely(ret)) if (unlikely(ret))
goto err_tx_restart;
if (whiteout) {
whiteout_inode_u = bch2_trans_kmalloc_nomemzero(trans, sizeof(*whiteout_inode_u));
ret = PTR_ERR_OR_ZERO(whiteout_inode_u);
if (unlikely(ret))
goto err_tx_restart;
bch2_inode_init_early(c, whiteout_inode_u);
ret = bch2_create_trans(trans,
inode_inum(src_dir), &src_dir_u,
whiteout_inode_u,
&src_dentry->d_name,
from_kuid(i_user_ns(&src_dir->v), current_fsuid()),
from_kgid(i_user_ns(&src_dir->v), current_fsgid()),
S_IFCHR|WHITEOUT_MODE, 0,
NULL, NULL, (subvol_inum) { 0 }, 0) ?:
bch2_quota_acct(c, bch_qid(whiteout_inode_u), Q_INO, 1,
KEY_TYPE_QUOTA_PREALLOC);
if (unlikely(ret))
goto err_tx_restart;
}
ret = bch2_trans_commit(trans, NULL, NULL, 0);
if (unlikely(ret)) {
err_tx_restart:
if (bch2_err_matches(ret, BCH_ERR_transaction_restart))
goto retry;
goto err; goto err;
}
BUG_ON(src_inode->v.i_ino != src_inode_u.bi_inum); BUG_ON(src_inode->v.i_ino != src_inode_u.bi_inum);
BUG_ON(dst_inode && BUG_ON(dst_inode &&
......
...@@ -270,7 +270,7 @@ int bch2_hash_set_in_snapshot(struct btree_trans *trans, ...@@ -270,7 +270,7 @@ int bch2_hash_set_in_snapshot(struct btree_trans *trans,
desc.hash_bkey(info, bkey_i_to_s_c(insert)), desc.hash_bkey(info, bkey_i_to_s_c(insert)),
snapshot), snapshot),
POS(insert->k.p.inode, U64_MAX), POS(insert->k.p.inode, U64_MAX),
BTREE_ITER_slots|BTREE_ITER_intent, k, ret) { BTREE_ITER_slots|BTREE_ITER_intent|flags, k, ret) {
if (is_visible_key(desc, inum, k)) { if (is_visible_key(desc, inum, k)) {
if (!desc.cmp_bkey(k, bkey_i_to_s_c(insert))) if (!desc.cmp_bkey(k, bkey_i_to_s_c(insert)))
goto found; goto found;
......
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