2017-03-17 09:18:50 +03:00
// SPDX-License-Identifier: GPL-2.0
# include "bcachefs.h"
2023-06-21 13:00:04 +03:00
# include "bkey_buf.h"
2017-03-17 09:18:50 +03:00
# include "bkey_methods.h"
# include "btree_update.h"
# include "extents.h"
# include "dirent.h"
# include "fs.h"
# include "keylist.h"
# include "str_hash.h"
2021-03-16 07:28:17 +03:00
# include "subvolume.h"
2017-03-17 09:18:50 +03:00
# include <linux/dcache.h>
2023-08-13 00:26:29 +03:00
static unsigned bch2_dirent_name_bytes ( struct bkey_s_c_dirent d )
2017-03-17 09:18:50 +03:00
{
2023-08-13 00:26:30 +03:00
unsigned bkey_u64s = bkey_val_u64s ( d . k ) ;
unsigned bkey_bytes = bkey_u64s * sizeof ( u64 ) ;
u64 last_u64 = ( ( u64 * ) d . v ) [ bkey_u64s - 1 ] ;
# if CPU_BIG_ENDIAN
unsigned trailing_nuls = last_u64 ? __builtin_ctzll ( last_u64 ) / 8 : 64 / 8 ;
# else
unsigned trailing_nuls = last_u64 ? __builtin_clzll ( last_u64 ) / 8 : 64 / 8 ;
# endif
return bkey_bytes -
offsetof ( struct bch_dirent , d_name ) -
trailing_nuls ;
2017-03-17 09:18:50 +03:00
}
2023-08-13 00:26:29 +03:00
struct qstr bch2_dirent_get_name ( struct bkey_s_c_dirent d )
{
return ( struct qstr ) QSTR_INIT ( d . v - > d_name , bch2_dirent_name_bytes ( d ) ) ;
}
2017-03-17 09:18:50 +03:00
static u64 bch2_dirent_hash ( const struct bch_hash_info * info ,
const struct qstr * name )
{
struct bch_str_hash_ctx ctx ;
bch2_str_hash_init ( & ctx , info ) ;
bch2_str_hash_update ( & ctx , info , name - > name , name - > len ) ;
/* [0,2) reserved for dots */
return max_t ( u64 , bch2_str_hash_end ( & ctx , info ) , 2 ) ;
}
static u64 dirent_hash_key ( const struct bch_hash_info * info , const void * key )
{
return bch2_dirent_hash ( info , key ) ;
}
static u64 dirent_hash_bkey ( const struct bch_hash_info * info , struct bkey_s_c k )
{
struct bkey_s_c_dirent d = bkey_s_c_to_dirent ( k ) ;
2023-08-13 00:26:29 +03:00
struct qstr name = bch2_dirent_get_name ( d ) ;
2017-03-17 09:18:50 +03:00
return bch2_dirent_hash ( info , & name ) ;
}
static bool dirent_cmp_key ( struct bkey_s_c _l , const void * _r )
{
struct bkey_s_c_dirent l = bkey_s_c_to_dirent ( _l ) ;
2023-08-13 00:26:29 +03:00
const struct qstr l_name = bch2_dirent_get_name ( l ) ;
const struct qstr * r_name = _r ;
2017-03-17 09:18:50 +03:00
2023-12-17 05:16:34 +03:00
return ! qstr_eq ( l_name , * r_name ) ;
2017-03-17 09:18:50 +03:00
}
static bool dirent_cmp_bkey ( struct bkey_s_c _l , struct bkey_s_c _r )
{
struct bkey_s_c_dirent l = bkey_s_c_to_dirent ( _l ) ;
struct bkey_s_c_dirent r = bkey_s_c_to_dirent ( _r ) ;
2023-08-13 00:26:29 +03:00
const struct qstr l_name = bch2_dirent_get_name ( l ) ;
const struct qstr r_name = bch2_dirent_get_name ( r ) ;
2017-03-17 09:18:50 +03:00
2023-12-17 05:16:34 +03:00
return ! qstr_eq ( l_name , r_name ) ;
2017-03-17 09:18:50 +03:00
}
2021-10-12 19:06:02 +03:00
static bool dirent_is_visible ( subvol_inum inum , struct bkey_s_c k )
{
struct bkey_s_c_dirent d = bkey_s_c_to_dirent ( k ) ;
if ( d . v - > d_type = = DT_SUBVOL )
return le32_to_cpu ( d . v - > d_parent_subvol ) = = inum . subvol ;
return true ;
}
2017-03-17 09:18:50 +03:00
const struct bch_hash_desc bch2_dirent_hash_desc = {
2021-02-21 03:27:37 +03:00
. btree_id = BTREE_ID_dirents ,
2018-11-01 22:10:01 +03:00
. key_type = KEY_TYPE_dirent ,
2017-03-17 09:18:50 +03:00
. hash_key = dirent_hash_key ,
. hash_bkey = dirent_hash_bkey ,
. cmp_key = dirent_cmp_key ,
. cmp_bkey = dirent_cmp_bkey ,
2021-10-12 19:06:02 +03:00
. is_visible = dirent_is_visible ,
2017-03-17 09:18:50 +03:00
} ;
2023-10-25 03:44:36 +03:00
int bch2_dirent_invalid ( struct bch_fs * c , struct bkey_s_c k ,
2023-07-07 04:16:10 +03:00
enum bkey_invalid_flags flags ,
struct printbuf * err )
2017-03-17 09:18:50 +03:00
{
2018-11-01 22:10:01 +03:00
struct bkey_s_c_dirent d = bkey_s_c_to_dirent ( k ) ;
2023-08-13 00:26:29 +03:00
struct qstr d_name = bch2_dirent_get_name ( d ) ;
2023-10-25 03:44:36 +03:00
int ret = 0 ;
2017-03-17 09:18:50 +03:00
2023-10-25 03:44:36 +03:00
bkey_fsck_err_on ( ! d_name . len , c , err ,
dirent_empty_name ,
" empty name " ) ;
2022-04-04 00:50:01 +03:00
2023-10-25 03:44:36 +03:00
bkey_fsck_err_on ( bkey_val_u64s ( k . k ) > dirent_val_u64s ( d_name . len ) , c , err ,
dirent_val_too_big ,
" value too big (%zu > %u) " ,
bkey_val_u64s ( k . k ) , dirent_val_u64s ( d_name . len ) ) ;
2022-04-04 00:50:01 +03:00
2023-08-13 18:53:45 +03:00
/*
* Check new keys don ' t exceed the max length
* ( older keys may be larger . )
*/
2023-10-25 03:44:36 +03:00
bkey_fsck_err_on ( ( flags & BKEY_INVALID_COMMIT ) & & d_name . len > BCH_NAME_MAX , c , err ,
dirent_name_too_long ,
" dirent name too big (%u > %u) " ,
d_name . len , BCH_NAME_MAX ) ;
bkey_fsck_err_on ( d_name . len ! = strnlen ( d_name . name , d_name . len ) , c , err ,
dirent_name_embedded_nul ,
" dirent has stray data after name's NUL " ) ;
bkey_fsck_err_on ( ( d_name . len = = 1 & & ! memcmp ( d_name . name , " . " , 1 ) ) | |
( d_name . len = = 2 & & ! memcmp ( d_name . name , " .. " , 2 ) ) , c , err ,
dirent_name_dot_or_dotdot ,
" invalid name " ) ;
bkey_fsck_err_on ( memchr ( d_name . name , ' / ' , d_name . len ) , c , err ,
dirent_name_has_slash ,
" name with / " ) ;
bkey_fsck_err_on ( d . v - > d_type ! = DT_SUBVOL & &
le64_to_cpu ( d . v - > d_inum ) = = d . k - > p . inode , c , err ,
dirent_to_itself ,
" dirent points to own directory " ) ;
fsck_err :
return ret ;
2017-03-17 09:18:50 +03:00
}
2024-01-22 01:46:14 +03:00
void bch2_dirent_to_text ( struct printbuf * out , struct bch_fs * c , struct bkey_s_c k )
2017-03-17 09:18:50 +03:00
{
2018-11-01 22:10:01 +03:00
struct bkey_s_c_dirent d = bkey_s_c_to_dirent ( k ) ;
2023-08-13 00:26:29 +03:00
struct qstr d_name = bch2_dirent_get_name ( d ) ;
2018-11-01 22:10:01 +03:00
2024-01-22 01:46:14 +03:00
prt_printf ( out , " %.*s -> " , d_name . len , d_name . name ) ;
if ( d . v - > d_type ! = DT_SUBVOL )
prt_printf ( out , " %llu " , le64_to_cpu ( d . v - > d_inum ) ) ;
else
prt_printf ( out , " %u -> %u " ,
le32_to_cpu ( d . v - > d_parent_subvol ) ,
le32_to_cpu ( d . v - > d_child_subvol ) ) ;
prt_printf ( out , " type %s " , bch2_d_type_str ( d . v - > d_type ) ) ;
2017-03-17 09:18:50 +03:00
}
static struct bkey_i_dirent * dirent_create_key ( struct btree_trans * trans ,
2021-10-12 19:06:02 +03:00
subvol_inum dir , u8 type ,
const struct qstr * name , u64 dst )
2017-03-17 09:18:50 +03:00
{
struct bkey_i_dirent * dirent ;
unsigned u64s = BKEY_U64s + dirent_val_u64s ( name - > len ) ;
if ( name - > len > BCH_NAME_MAX )
return ERR_PTR ( - ENAMETOOLONG ) ;
BUG_ON ( u64s > U8_MAX ) ;
dirent = bch2_trans_kmalloc ( trans , u64s * sizeof ( u64 ) ) ;
if ( IS_ERR ( dirent ) )
return dirent ;
bkey_dirent_init ( & dirent - > k_i ) ;
dirent - > k . u64s = u64s ;
2021-10-12 19:06:02 +03:00
if ( type ! = DT_SUBVOL ) {
dirent - > v . d_inum = cpu_to_le64 ( dst ) ;
} else {
dirent - > v . d_parent_subvol = cpu_to_le32 ( dir . subvol ) ;
dirent - > v . d_child_subvol = cpu_to_le32 ( dst ) ;
}
2017-03-17 09:18:50 +03:00
dirent - > v . d_type = type ;
memcpy ( dirent - > v . d_name , name - > name , name - > len ) ;
memset ( dirent - > v . d_name + name - > len , 0 ,
bkey_val_bytes ( & dirent - > k ) -
offsetof ( struct bch_dirent , d_name ) -
name - > len ) ;
EBUG_ON ( bch2_dirent_name_bytes ( dirent_i_to_s_c ( dirent ) ) ! = name - > len ) ;
return dirent ;
}
2023-12-15 22:13:48 +03:00
int bch2_dirent_create_snapshot ( struct btree_trans * trans ,
2024-02-10 00:04:50 +03:00
u32 dir_subvol , u64 dir , u32 snapshot ,
2023-12-15 22:13:48 +03:00
const struct bch_hash_info * hash_info ,
u8 type , const struct qstr * name , u64 dst_inum ,
u64 * dir_offset ,
bch_str_hash_flags_t str_hash_flags )
{
2024-02-10 00:04:50 +03:00
subvol_inum dir_inum = { . subvol = dir_subvol , . inum = dir } ;
2023-12-15 22:13:48 +03:00
struct bkey_i_dirent * dirent ;
int ret ;
2024-02-10 00:04:50 +03:00
dirent = dirent_create_key ( trans , dir_inum , type , name , dst_inum ) ;
2023-12-15 22:13:48 +03:00
ret = PTR_ERR_OR_ZERO ( dirent ) ;
if ( ret )
return ret ;
dirent - > k . p . inode = dir ;
dirent - > k . p . snapshot = snapshot ;
2024-01-26 03:57:26 +03:00
ret = bch2_hash_set_in_snapshot ( trans , bch2_dirent_hash_desc , hash_info ,
2024-02-10 00:04:50 +03:00
dir_inum , snapshot ,
2024-01-26 03:57:26 +03:00
& dirent - > k_i , str_hash_flags ,
BTREE_UPDATE_INTERNAL_SNAPSHOT_NODE ) ;
2023-12-15 22:13:48 +03:00
* dir_offset = dirent - > k . p . offset ;
return ret ;
}
2021-03-16 07:28:17 +03:00
int bch2_dirent_create ( struct btree_trans * trans , subvol_inum dir ,
const struct bch_hash_info * hash_info ,
2019-10-03 01:35:36 +03:00
u8 type , const struct qstr * name , u64 dst_inum ,
2023-11-12 00:31:50 +03:00
u64 * dir_offset ,
bch_str_hash_flags_t str_hash_flags )
2017-03-17 09:18:50 +03:00
{
struct bkey_i_dirent * dirent ;
int ret ;
2021-10-12 19:06:02 +03:00
dirent = dirent_create_key ( trans , dir , type , name , dst_inum ) ;
2017-03-17 09:18:50 +03:00
ret = PTR_ERR_OR_ZERO ( dirent ) ;
if ( ret )
return ret ;
2021-03-03 02:35:30 +03:00
ret = bch2_hash_set ( trans , bch2_dirent_hash_desc , hash_info ,
2023-11-12 00:31:50 +03:00
dir , & dirent - > k_i , str_hash_flags ) ;
2021-03-03 02:35:30 +03:00
* dir_offset = dirent - > k . p . offset ;
return ret ;
2017-03-17 09:18:50 +03:00
}
static void dirent_copy_target ( struct bkey_i_dirent * dst ,
struct bkey_s_c_dirent src )
{
dst - > v . d_inum = src . v - > d_inum ;
dst - > v . d_type = src . v - > d_type ;
}
2021-11-14 03:49:14 +03:00
int bch2_dirent_read_target ( struct btree_trans * trans , subvol_inum dir ,
struct bkey_s_c_dirent d , subvol_inum * target )
2021-03-16 07:46:26 +03:00
{
2021-10-14 20:14:40 +03:00
struct bch_subvolume s ;
2021-03-16 07:46:26 +03:00
int ret = 0 ;
2021-10-12 19:06:02 +03:00
if ( d . v - > d_type = = DT_SUBVOL & &
2023-07-07 05:47:42 +03:00
le32_to_cpu ( d . v - > d_parent_subvol ) ! = dir . subvol )
2021-10-12 19:06:02 +03:00
return 1 ;
2021-03-16 07:46:26 +03:00
if ( likely ( d . v - > d_type ! = DT_SUBVOL ) ) {
2021-10-12 19:06:02 +03:00
target - > subvol = dir . subvol ;
target - > inum = le64_to_cpu ( d . v - > d_inum ) ;
2021-03-16 07:46:26 +03:00
} else {
2021-10-12 19:06:02 +03:00
target - > subvol = le32_to_cpu ( d . v - > d_child_subvol ) ;
2021-10-01 02:46:23 +03:00
2021-10-12 19:06:02 +03:00
ret = bch2_subvolume_get ( trans , target - > subvol , true , BTREE_ITER_CACHED , & s ) ;
2021-10-01 02:46:23 +03:00
2021-10-12 19:06:02 +03:00
target - > inum = le64_to_cpu ( s . inode ) ;
2021-03-16 07:46:26 +03:00
}
return ret ;
}
2017-03-17 09:18:50 +03:00
int bch2_dirent_rename ( struct btree_trans * trans ,
2021-03-16 07:28:17 +03:00
subvol_inum src_dir , struct bch_hash_info * src_hash ,
subvol_inum dst_dir , struct bch_hash_info * dst_hash ,
const struct qstr * src_name , subvol_inum * src_inum , u64 * src_offset ,
const struct qstr * dst_name , subvol_inum * dst_inum , u64 * dst_offset ,
enum bch_rename_mode mode )
2017-03-17 09:18:50 +03:00
{
2021-08-30 22:18:31 +03:00
struct btree_iter src_iter = { NULL } ;
struct btree_iter dst_iter = { NULL } ;
2021-12-15 00:05:47 +03:00
struct bkey_s_c old_src , old_dst = bkey_s_c_null ;
2017-03-17 09:18:50 +03:00
struct bkey_i_dirent * new_src = NULL , * new_dst = NULL ;
2019-10-03 01:35:36 +03:00
struct bpos dst_pos =
2021-03-16 07:28:17 +03:00
POS ( dst_dir . inum , bch2_dirent_hash ( dst_hash , dst_name ) ) ;
2024-01-22 00:46:45 +03:00
unsigned src_update_flags = 0 ;
bool delete_src , delete_dst ;
2020-02-26 23:39:46 +03:00
int ret = 0 ;
2017-03-17 09:18:50 +03:00
2021-03-16 07:28:17 +03:00
memset ( src_inum , 0 , sizeof ( * src_inum ) ) ;
memset ( dst_inum , 0 , sizeof ( * dst_inum ) ) ;
2019-10-03 01:35:36 +03:00
2017-03-17 09:18:50 +03:00
/* Lookup src: */
2021-08-30 22:18:31 +03:00
ret = bch2_hash_lookup ( trans , & src_iter , bch2_dirent_hash_desc ,
src_hash , src_dir , src_name ,
BTREE_ITER_INTENT ) ;
2020-02-26 23:39:46 +03:00
if ( ret )
goto out ;
2021-08-30 22:18:31 +03:00
old_src = bch2_btree_iter_peek_slot ( & src_iter ) ;
2021-07-25 02:50:40 +03:00
ret = bkey_err ( old_src ) ;
if ( ret )
goto out ;
2021-03-16 07:28:17 +03:00
ret = bch2_dirent_read_target ( trans , src_dir ,
bkey_s_c_to_dirent ( old_src ) , src_inum ) ;
if ( ret )
goto out ;
2017-03-17 09:18:50 +03:00
2021-10-12 19:06:02 +03:00
/* Lookup dst: */
if ( mode = = BCH_RENAME ) {
/*
* Note that we ' re _not_ checking if the target already exists -
* we ' re relying on the VFS to do that check for us for
* correctness :
*/
ret = bch2_hash_hole ( trans , & dst_iter , bch2_dirent_hash_desc ,
dst_hash , dst_dir , dst_name ) ;
if ( ret )
goto out ;
} else {
ret = bch2_hash_lookup ( trans , & dst_iter , bch2_dirent_hash_desc ,
dst_hash , dst_dir , dst_name ,
BTREE_ITER_INTENT ) ;
if ( ret )
goto out ;
old_dst = bch2_btree_iter_peek_slot ( & dst_iter ) ;
ret = bkey_err ( old_dst ) ;
if ( ret )
goto out ;
ret = bch2_dirent_read_target ( trans , dst_dir ,
bkey_s_c_to_dirent ( old_dst ) , dst_inum ) ;
if ( ret )
goto out ;
}
if ( mode ! = BCH_RENAME_EXCHANGE )
* src_offset = dst_iter . pos . offset ;
2017-03-17 09:18:50 +03:00
/* Create new dst key: */
2021-10-12 19:06:02 +03:00
new_dst = dirent_create_key ( trans , dst_dir , 0 , dst_name , 0 ) ;
2020-02-26 23:39:46 +03:00
ret = PTR_ERR_OR_ZERO ( new_dst ) ;
if ( ret )
goto out ;
2017-03-17 09:18:50 +03:00
dirent_copy_target ( new_dst , bkey_s_c_to_dirent ( old_src ) ) ;
2021-08-30 22:18:31 +03:00
new_dst - > k . p = dst_iter . pos ;
2017-03-17 09:18:50 +03:00
/* Create new src key: */
if ( mode = = BCH_RENAME_EXCHANGE ) {
2021-10-12 19:06:02 +03:00
new_src = dirent_create_key ( trans , src_dir , 0 , src_name , 0 ) ;
2020-02-26 23:39:46 +03:00
ret = PTR_ERR_OR_ZERO ( new_src ) ;
if ( ret )
goto out ;
2017-03-17 09:18:50 +03:00
dirent_copy_target ( new_src , bkey_s_c_to_dirent ( old_dst ) ) ;
2021-08-30 22:18:31 +03:00
new_src - > k . p = src_iter . pos ;
2017-03-17 09:18:50 +03:00
} else {
new_src = bch2_trans_kmalloc ( trans , sizeof ( struct bkey_i ) ) ;
2020-02-26 23:39:46 +03:00
ret = PTR_ERR_OR_ZERO ( new_src ) ;
if ( ret )
goto out ;
2017-03-17 09:18:50 +03:00
bkey_init ( & new_src - > k ) ;
2021-08-30 22:18:31 +03:00
new_src - > k . p = src_iter . pos ;
2017-03-17 09:18:50 +03:00
2022-11-24 11:12:22 +03:00
if ( bkey_le ( dst_pos , src_iter . pos ) & &
bkey_lt ( src_iter . pos , dst_iter . pos ) ) {
2017-03-17 09:18:50 +03:00
/*
* We have a hash collision for the new dst key ,
* and new_src - the key we ' re deleting - is between
* new_dst ' s hashed slot and the slot we ' re going to be
* inserting it into - oops . This will break the hash
* table if we don ' t deal with it :
*/
if ( mode = = BCH_RENAME ) {
/*
* If we ' re not overwriting , we can just insert
* new_dst at the src position :
*/
2021-10-12 19:06:02 +03:00
new_src = new_dst ;
new_src - > k . p = src_iter . pos ;
goto out_set_src ;
2017-03-17 09:18:50 +03:00
} else {
/* If we're overwriting, we can't insert new_dst
* at a different slot because it has to
* overwrite old_dst - just make sure to use a
* whiteout when deleting src :
*/
2021-02-21 03:09:53 +03:00
new_src - > k . type = KEY_TYPE_hash_whiteout ;
2017-03-17 09:18:50 +03:00
}
} else {
/* Check if we need a whiteout to delete src: */
ret = bch2_hash_needs_whiteout ( trans , bch2_dirent_hash_desc ,
2021-08-30 22:18:31 +03:00
src_hash , & src_iter ) ;
2017-03-17 09:18:50 +03:00
if ( ret < 0 )
2020-02-26 23:39:46 +03:00
goto out ;
2017-03-17 09:18:50 +03:00
if ( ret )
2021-02-21 03:09:53 +03:00
new_src - > k . type = KEY_TYPE_hash_whiteout ;
2017-03-17 09:18:50 +03:00
}
}
2024-01-22 00:46:45 +03:00
if ( new_dst - > v . d_type = = DT_SUBVOL )
new_dst - > v . d_parent_subvol = cpu_to_le32 ( dst_dir . subvol ) ;
if ( ( mode = = BCH_RENAME_EXCHANGE ) & &
new_src - > v . d_type = = DT_SUBVOL )
new_src - > v . d_parent_subvol = cpu_to_le32 ( src_dir . subvol ) ;
2021-12-05 08:30:49 +03:00
ret = bch2_trans_update ( trans , & dst_iter , & new_dst - > k_i , 0 ) ;
if ( ret )
goto out ;
2021-10-12 19:06:02 +03:00
out_set_src :
/*
2024-01-22 00:46:45 +03:00
* If we ' re deleting a subvolume we need to really delete the dirent ,
* not just emit a whiteout in the current snapshot - there can only be
* single dirent that points to a given subvolume .
*
* IOW , we don ' t maintain multiple versions in different snapshots of
* dirents that point to subvolumes - dirents that point to subvolumes
* are only visible in one particular subvolume so it ' s not necessary ,
* and it would be particularly confusing for fsck to have to deal with .
2021-10-12 19:06:02 +03:00
*/
2024-01-22 00:46:45 +03:00
delete_src = bkey_s_c_to_dirent ( old_src ) . v - > d_type = = DT_SUBVOL & &
new_src - > k . p . snapshot ! = old_src . k - > p . snapshot ;
delete_dst = old_dst . k & &
bkey_s_c_to_dirent ( old_dst ) . v - > d_type = = DT_SUBVOL & &
new_dst - > k . p . snapshot ! = old_dst . k - > p . snapshot ;
if ( ! delete_src | | ! bkey_deleted ( & new_src - > k ) ) {
ret = bch2_trans_update ( trans , & src_iter , & new_src - > k_i , src_update_flags ) ;
2021-10-12 19:06:02 +03:00
if ( ret )
goto out ;
2024-01-22 00:46:45 +03:00
}
2021-10-12 19:06:02 +03:00
2024-01-22 00:46:45 +03:00
if ( delete_src ) {
bch2_btree_iter_set_snapshot ( & src_iter , old_src . k - > p . snapshot ) ;
ret = bch2_btree_iter_traverse ( & src_iter ) ? :
bch2_btree_delete_at ( trans , & src_iter , BTREE_UPDATE_INTERNAL_SNAPSHOT_NODE ) ;
if ( ret )
goto out ;
2021-10-12 19:06:02 +03:00
}
2024-01-22 00:46:45 +03:00
if ( delete_dst ) {
bch2_btree_iter_set_snapshot ( & dst_iter , old_dst . k - > p . snapshot ) ;
ret = bch2_btree_iter_traverse ( & dst_iter ) ? :
bch2_btree_delete_at ( trans , & dst_iter , BTREE_UPDATE_INTERNAL_SNAPSHOT_NODE ) ;
if ( ret )
goto out ;
}
2021-10-12 19:06:02 +03:00
2021-05-20 07:09:47 +03:00
if ( mode = = BCH_RENAME_EXCHANGE )
* src_offset = new_src - > k . p . offset ;
2021-03-03 02:35:30 +03:00
* dst_offset = new_dst - > k . p . offset ;
2020-02-26 23:39:46 +03:00
out :
2021-08-30 22:18:31 +03:00
bch2_trans_iter_exit ( trans , & src_iter ) ;
bch2_trans_iter_exit ( trans , & dst_iter ) ;
2020-02-26 23:39:46 +03:00
return ret ;
2017-03-17 09:18:50 +03:00
}
2024-01-25 20:35:06 +03:00
int bch2_dirent_lookup_trans ( struct btree_trans * trans ,
struct btree_iter * iter ,
subvol_inum dir ,
const struct bch_hash_info * hash_info ,
const struct qstr * name , subvol_inum * inum ,
unsigned flags )
2019-10-03 01:35:36 +03:00
{
2024-01-25 20:35:06 +03:00
int ret = bch2_hash_lookup ( trans , iter , bch2_dirent_hash_desc ,
hash_info , dir , name , flags ) ;
2021-03-16 07:46:26 +03:00
if ( ret )
return ret ;
2024-01-25 20:35:06 +03:00
struct bkey_s_c k = bch2_btree_iter_peek_slot ( iter ) ;
2021-03-16 07:46:26 +03:00
ret = bkey_err ( k ) ;
2021-11-13 20:57:00 +03:00
if ( ret )
goto err ;
2021-03-16 07:46:26 +03:00
2024-01-25 20:35:06 +03:00
ret = bch2_dirent_read_target ( trans , dir , bkey_s_c_to_dirent ( k ) , inum ) ;
2021-10-12 19:06:02 +03:00
if ( ret > 0 )
ret = - ENOENT ;
2021-11-13 20:57:00 +03:00
err :
2021-03-16 07:46:26 +03:00
if ( ret )
bch2_trans_iter_exit ( trans , iter ) ;
return ret ;
2017-03-17 09:18:50 +03:00
}
2021-03-16 07:28:17 +03:00
u64 bch2_dirent_lookup ( struct bch_fs * c , subvol_inum dir ,
2017-03-17 09:18:50 +03:00
const struct bch_hash_info * hash_info ,
2021-03-16 07:28:17 +03:00
const struct qstr * name , subvol_inum * inum )
2017-03-17 09:18:50 +03:00
{
2023-09-13 00:16:02 +03:00
struct btree_trans * trans = bch2_trans_get ( c ) ;
2023-12-17 11:05:30 +03:00
struct btree_iter iter = { NULL } ;
int ret = lockrestart_do ( trans ,
2024-01-25 20:35:06 +03:00
bch2_dirent_lookup_trans ( trans , & iter , dir , hash_info , name , inum , 0 ) ) ;
2023-12-17 11:05:30 +03:00
bch2_trans_iter_exit ( trans , & iter ) ;
2023-09-13 00:16:02 +03:00
bch2_trans_put ( trans ) ;
2021-03-16 07:28:17 +03:00
return ret ;
2017-03-17 09:18:50 +03:00
}
2024-02-13 11:20:04 +03:00
int bch2_empty_dir_snapshot ( struct btree_trans * trans , u64 dir , u32 subvol , u32 snapshot )
2017-03-17 09:18:50 +03:00
{
2021-08-30 22:18:31 +03:00
struct btree_iter iter ;
2017-03-17 09:18:50 +03:00
struct bkey_s_c k ;
2019-04-17 22:49:28 +03:00
int ret ;
2019-03-25 22:10:15 +03:00
2022-03-11 20:31:52 +03:00
for_each_btree_key_upto_norestart ( trans , iter , BTREE_ID_dirents ,
2023-12-07 20:39:13 +03:00
SPOS ( dir , 0 , snapshot ) ,
POS ( dir , U64_MAX ) , 0 , k , ret )
2018-11-01 22:10:01 +03:00
if ( k . k - > type = = KEY_TYPE_dirent ) {
2024-02-13 11:20:04 +03:00
struct bkey_s_c_dirent d = bkey_s_c_to_dirent ( k ) ;
if ( d . v - > d_type = = DT_SUBVOL & & le32_to_cpu ( d . v - > d_parent_subvol ) ! = subvol )
continue ;
2024-02-10 05:01:04 +03:00
ret = - BCH_ERR_ENOTEMPTY_dir_not_empty ;
2017-03-17 09:18:50 +03:00
break ;
}
2021-08-30 22:18:31 +03:00
bch2_trans_iter_exit ( trans , & iter ) ;
2017-03-17 09:18:50 +03:00
return ret ;
}
2023-12-07 20:39:13 +03:00
int bch2_empty_dir_trans ( struct btree_trans * trans , subvol_inum dir )
{
u32 snapshot ;
return bch2_subvolume_get_snapshot ( trans , dir . subvol , & snapshot ) ? :
2024-02-13 11:20:04 +03:00
bch2_empty_dir_snapshot ( trans , dir . inum , dir . subvol , snapshot ) ;
2023-12-07 20:39:13 +03:00
}
2021-03-16 07:28:17 +03:00
int bch2_readdir ( struct bch_fs * c , subvol_inum inum , struct dir_context * ctx )
2019-04-01 00:37:30 +03:00
{
2023-09-13 00:16:02 +03:00
struct btree_trans * trans = bch2_trans_get ( c ) ;
2021-08-30 22:18:31 +03:00
struct btree_iter iter ;
2017-03-17 09:18:50 +03:00
struct bkey_s_c k ;
struct bkey_s_c_dirent dirent ;
2021-10-12 19:06:02 +03:00
subvol_inum target ;
2021-03-16 07:28:17 +03:00
u32 snapshot ;
2023-06-21 13:00:04 +03:00
struct bkey_buf sk ;
2023-08-13 00:26:29 +03:00
struct qstr name ;
2019-04-17 22:49:28 +03:00
int ret ;
2017-03-17 09:18:50 +03:00
2023-06-21 13:00:04 +03:00
bch2_bkey_buf_init ( & sk ) ;
2021-03-16 07:28:17 +03:00
retry :
2023-09-13 00:16:02 +03:00
bch2_trans_begin ( trans ) ;
2021-03-16 07:28:17 +03:00
2023-09-13 00:16:02 +03:00
ret = bch2_subvolume_get_snapshot ( trans , inum . subvol , & snapshot ) ;
2021-03-16 07:28:17 +03:00
if ( ret )
goto err ;
2019-03-25 22:10:15 +03:00
2023-09-13 00:16:02 +03:00
for_each_btree_key_upto_norestart ( trans , iter , BTREE_ID_dirents ,
2022-03-11 20:31:52 +03:00
SPOS ( inum . inum , ctx - > pos , snapshot ) ,
POS ( inum . inum , U64_MAX ) , 0 , k , ret ) {
2018-11-01 22:10:01 +03:00
if ( k . k - > type ! = KEY_TYPE_dirent )
2017-03-17 09:18:50 +03:00
continue ;
dirent = bkey_s_c_to_dirent ( k ) ;
2023-09-13 00:16:02 +03:00
ret = bch2_dirent_read_target ( trans , inum , dirent , & target ) ;
2021-10-12 19:06:02 +03:00
if ( ret < 0 )
break ;
if ( ret )
continue ;
2023-06-21 13:00:04 +03:00
/* dir_emit() can fault and block: */
bch2_bkey_buf_reassemble ( & sk , c , k ) ;
dirent = bkey_i_to_s_c_dirent ( sk . k ) ;
2023-09-13 00:16:02 +03:00
bch2_trans_unlock ( trans ) ;
2023-06-21 13:00:04 +03:00
2023-08-13 00:26:29 +03:00
name = bch2_dirent_get_name ( dirent ) ;
2019-10-09 16:23:30 +03:00
ctx - > pos = dirent . k - > p . offset ;
2023-08-13 00:26:29 +03:00
if ( ! dir_emit ( ctx , name . name ,
name . len ,
2021-10-12 19:06:02 +03:00
target . inum ,
2021-03-16 07:46:26 +03:00
vfs_d_type ( dirent . v - > d_type ) ) )
2017-03-17 09:18:50 +03:00
break ;
2019-10-09 16:23:30 +03:00
ctx - > pos = dirent . k - > p . offset + 1 ;
2021-10-23 00:33:38 +03:00
/*
* read_target looks up subvolumes , we can overflow paths if the
* directory has many subvolumes in it
*/
2023-09-13 00:16:02 +03:00
ret = btree_trans_too_many_iters ( trans ) ;
2021-11-24 03:00:23 +03:00
if ( ret )
2021-10-23 00:33:38 +03:00
break ;
2017-03-17 09:18:50 +03:00
}
2023-09-13 00:16:02 +03:00
bch2_trans_iter_exit ( trans , & iter ) ;
2021-03-16 07:28:17 +03:00
err :
2022-07-18 06:06:38 +03:00
if ( bch2_err_matches ( ret , BCH_ERR_transaction_restart ) )
2021-03-16 07:28:17 +03:00
goto retry ;
2021-03-20 03:29:11 +03:00
2023-09-13 00:16:02 +03:00
bch2_trans_put ( trans ) ;
2023-06-21 13:00:04 +03:00
bch2_bkey_buf_exit ( & sk , c ) ;
2017-03-17 09:18:50 +03:00
2019-04-17 22:49:28 +03:00
return ret ;
2017-03-17 09:18:50 +03:00
}