* Questions and notes about send/receive
@ 2012-07-31 16:32 Alex Lyakas
2012-08-01 11:10 ` Alexander Block
0 siblings, 1 reply; 3+ messages in thread
From: Alex Lyakas @ 2012-07-31 16:32 UTC (permalink / raw)
To: Alexander Block; +Cc: linux-btrfs
Hi Alexander,
I relooked at my list of questions, and it seems that there are more
general questions, and more focused questions. So here I list the
"more focused" ones. I really appreciate if you can address them. I
hope it's ok that I opened a separate thread for them. So here goes...
iterate_dir_item(): tmp_path is not used
name_cache_insert(): if radix_tree_insert() fails, should we free the nce_head?
__get_cur_name_and_parent()
if name_cache_insert() fails, should we free the nce?
__get_cur_name_and_parent()
if is_inode_existent() is false, and we generate unique name, we don't
set *parent_ino/*parent_gen....I guess it's ok, it just looked strange
when it showed up in my prints.
btrfs_ioctl_send()
struct file *filp = NULL; => this is not used
btrfs_ioctl_set_received_subvol()
if we fail to commit the transaction, should we rollback the in-memory
values that we set, like stransid/rtransid etc? or they get rolled
back automatically somehow (I need to study the transaction
subsystem).
process_recorded_refs():
/*
* If the inode is still orphan, unlink the orphan. This may
* happen when a previous inode did overwrite the first ref
* of this inode and no new refs were added for the current
* inode.
*/
This comment is partially misleading, I think. It implies that the
orphan inode will be deleted. But it may happen that it will only be
unlinked, because another hardlink to it remains (one of my tests goes
through this path - inode is orphanized, then orphan unlinked, but
another hardlink exists).
btrfs-progs::read_and_process_cmd()
if (strstr(path, ".bak_1.log")) {
ret = 0;
}
this block is some leftover?
__get_cur_name_and_parent()
if we decided to call get_first_ref() on the send_root (due to ino <
sctx->send_progress), do we really need to call did_overwrite_ref()?
Because it will just lookup the send root again. I mean if we know
that this inode has been already handled, then it's not an orphan
anymore, so no need for this additional check. If my understanding is
wrong, pls give some small example?
process_recorded_refs():
why we need to call did_overwrite_first_ref(), and not just call
get_cur_path()? It looks like we need this only to set the is_orphan
flag? Because, otherwise, get_cur_path() already does the
overwrite_first_ref logic, and will return the correct path. Is my
understanding correct? (I ran some tests and looks correct to me).
find_extent_clone():
/*
* Now collect all backrefs.
*/
extent_item_pos = logical - found_key.objectid;
Is my understanding correct that extent_item_pos will now be equal to
btrfs_file_extent_offset(eb, fi)?
process_recorded_refs():
if (ret == inode_state_did_create ||
ret == inode_state_no_change) {
/* TODO delayed utimes */
ret = send_utimes(sctx, un->val, un->aux);
This code results in UTIMES sent twice for a parent dir of new inode:
once because parent_dir gets changed (stamped with a transid) and
second time because of this code. Is this fine? Also what "TODO
delayed utimes" means?
process_recorded_refs():
if (un->val > sctx->cur_ino)
continue;
This skips some directories. Is this because we will anyhow attend to
them later because of their ino?
is_extent_unchanged()
/*
* Check if we have the same extent.
*/
if (left_disknr + left_offset_fixed !=
right_disknr + right_offset) {
ret = 0;
goto out;
}
should also check: left_disknr == right_disknr for extra safety?
Because disknr is what really defines the extent.
is_extent_unchanged():
/*
* Are we at extent 8? If yes, we know the extent is changed.
* This may only happen on the first iteration.
*/
if (found_key.offset + right_len < ekey->offset) {
ret = 0;
goto out;
}
Should it be:
if (found_key.offset + right_len <= ekey->offset) {
?
Because in this case right extent is not overlapping also.
struct btrfs_root_item:
what is the difference between existing "generation" field that you
mimic in generation_v2 and "ctransid"? (I know I need to study the
root tree code).
Thank you,
Alex.
^ permalink raw reply [flat|nested] 3+ messages in thread
* Re: Questions and notes about send/receive
2012-07-31 16:32 Questions and notes about send/receive Alex Lyakas
@ 2012-08-01 11:10 ` Alexander Block
2012-08-01 17:31 ` Alex Lyakas
0 siblings, 1 reply; 3+ messages in thread
From: Alexander Block @ 2012-08-01 11:10 UTC (permalink / raw)
To: Alex Lyakas; +Cc: linux-btrfs
On Tue, Jul 31, 2012 at 6:32 PM, Alex Lyakas
<alex.bolshoy.btrfs@gmail.com> wrote:
> Hi Alexander,
> I relooked at my list of questions, and it seems that there are more
> general questions, and more focused questions. So here I list the
> "more focused" ones. I really appreciate if you can address them. I
> hope it's ok that I opened a separate thread for them. So here goes...
>
> iterate_dir_item(): tmp_path is not used
Removed.
>
> name_cache_insert(): if radix_tree_insert() fails, should we free the nce_head?
>
> __get_cur_name_and_parent()
> if name_cache_insert() fails, should we free the nce?
Both, nce_head and nce are now freed in name_cache_insert.
>
> __get_cur_name_and_parent()
> if is_inode_existent() is false, and we generate unique name, we don't
> set *parent_ino/*parent_gen....I guess it's ok, it just looked strange
> when it showed up in my prints.
Yepp, that's ok. We don't have a valid parent ino/gen in that case. I
added a note to the function comment.
>
> btrfs_ioctl_send()
> struct file *filp = NULL; => this is not used
>
> btrfs_ioctl_set_received_subvol()
> if we fail to commit the transaction, should we rollback the in-memory
> values that we set, like stransid/rtransid etc? or they get rolled
> back automatically somehow (I need to study the transaction
> subsystem).
Hmm not sure about that. In case of transaction abort, the FS is not
writable anymore as far as I know, so no matter which values are
in-memory, the user won't be able to use them.
>
> process_recorded_refs():
> /*
> * If the inode is still orphan, unlink the orphan. This may
> * happen when a previous inode did overwrite the first ref
> * of this inode and no new refs were added for the current
> * inode.
> */
> This comment is partially misleading, I think. It implies that the
> orphan inode will be deleted. But it may happen that it will only be
> unlinked, because another hardlink to it remains (one of my tests goes
> through this path - inode is orphanized, then orphan unlinked, but
> another hardlink exists).
This behavior is expected. We do the orphan handling only on first
refs and never check if that may generate unnecessary commands. I made
it this way because it was the easiest and most secure solution, but
it has room for optimization. A added a note about this to the
comment.
>
> btrfs-progs::read_and_process_cmd()
> if (strstr(path, ".bak_1.log")) {
> ret = 0;
> }
> this block is some leftover?
Hehe, yes :) I love such statements so that I can add a breakpoint to
"ret=0" :) I removed it.
>
> __get_cur_name_and_parent()
> if we decided to call get_first_ref() on the send_root (due to ino <
> sctx->send_progress), do we really need to call did_overwrite_ref()?
> Because it will just lookup the send root again. I mean if we know
> that this inode has been already handled, then it's not an orphan
> anymore, so no need for this additional check. If my understanding is
> wrong, pls give some small example?
get_first_ref does not return the current state. It returns the
permanent state. The result of did_overwrite_ref however depends on
the current send progress. There is however some room for optimization
here. In many cases, I did not think much about double tree lookups
due to the different calls, as I wanted it to work first and later
optimize it. One possible optimization for example would be to cache
the results of lookups. But only for functions which return permanent
state, e.g. get_inode_info.
>
> process_recorded_refs():
> why we need to call did_overwrite_first_ref(), and not just call
> get_cur_path()? It looks like we need this only to set the is_orphan
> flag? Because, otherwise, get_cur_path() already does the
> overwrite_first_ref logic, and will return the correct path. Is my
> understanding correct? (I ran some tests and looks correct to me).
Hmm this is probably true. It was probably required in the past to
call did_overwrite_first_ref, not sure. I would like to keep it this
way for the moment as I don't want to risk introducing new bugs.
>
> find_extent_clone():
> /*
> * Now collect all backrefs.
> */
> extent_item_pos = logical - found_key.objectid;
> Is my understanding correct that extent_item_pos will now be equal to
> btrfs_file_extent_offset(eb, fi)?
Yepp. It's probably not needed to do it this way, but I left it this
way as this part of the code is copied from Jan's initial version of
btrfs send.
>
> process_recorded_refs():
> if (ret == inode_state_did_create ||
> ret == inode_state_no_change) {
> /* TODO delayed utimes */
> ret = send_utimes(sctx, un->val, un->aux);
> This code results in UTIMES sent twice for a parent dir of new inode:
> once because parent_dir gets changed (stamped with a transid) and
> second time because of this code. Is this fine? Also what "TODO
> delayed utimes" means?
I'm collection TODOs here:
https://btrfs.wiki.kernel.org/index.php/Btrfs_Send/Receive
One of the TODOs describes the delayed utime updates.
>
> process_recorded_refs():
> if (un->val > sctx->cur_ino)
> continue;
> This skips some directories. Is this because we will anyhow attend to
> them later because of their ino?
Yepp. Added a comment about this.
>
> is_extent_unchanged()
> /*
> * Check if we have the same extent.
> */
> if (left_disknr + left_offset_fixed !=
> right_disknr + right_offset) {
> ret = 0;
> goto out;
> }
> should also check: left_disknr == right_disknr for extra safety?
> Because disknr is what really defines the extent.
I changed it to:
if (left_disknr != right_disknr ||
left_offset_fixed != right_offset) {
Should be more clear...
>
> is_extent_unchanged():
> /*
> * Are we at extent 8? If yes, we know the extent is changed.
> * This may only happen on the first iteration.
> */
> if (found_key.offset + right_len < ekey->offset) {
> ret = 0;
> goto out;
> }
> Should it be:
> if (found_key.offset + right_len <= ekey->offset) {
> ?
> Because in this case right extent is not overlapping also.
Hmm, true. Fixed that.
>
> struct btrfs_root_item:
> what is the difference between existing "generation" field that you
> mimic in generation_v2 and "ctransid"? (I know I need to study the
> root tree code).
Did you read the comments for btrfs_root_item and
btrfs_read_root_item? They should answer your question.
>
> Thank you,
> Alex.
Big thanks for your reviews :)
I pushed a new version of the kernel and progs side. The branches are
now "for-chris"
^ permalink raw reply [flat|nested] 3+ messages in thread
* Re: Questions and notes about send/receive
2012-08-01 11:10 ` Alexander Block
@ 2012-08-01 17:31 ` Alex Lyakas
0 siblings, 0 replies; 3+ messages in thread
From: Alex Lyakas @ 2012-08-01 17:31 UTC (permalink / raw)
To: Alexander Block; +Cc: linux-btrfs
Alexander,
thanks for addressing the issues.
>> __get_cur_name_and_parent()
>> if we decided to call get_first_ref() on the send_root (due to ino <
>> sctx->send_progress), do we really need to call did_overwrite_ref()?
>> Because it will just lookup the send root again. I mean if we know
>> that this inode has been already handled, then it's not an orphan
>> anymore, so no need for this additional check. If my understanding is
>> wrong, pls give some small example?
> get_first_ref does not return the current state. It returns the
> permanent state. The result of did_overwrite_ref however depends on
> the current send progress. There is however some room for optimization
> here. In many cases, I did not think much about double tree lookups
> due to the different calls, as I wanted it to work first and later
> optimize it. One possible optimization for example would be to cache
> the results of lookups. But only for functions which return permanent
> state, e.g. get_inode_info.
So do you think my understanding is correct that if (ino <
sctx->send_progress), then both functions will return the same value?
Or perhaps: if (ino < sctx->send_progress), then no need to check
anymore the did_overwrite_ref(), because we have handled that already?
I think that the call to did_overwrite_ref() is relevant only as long
as we haven't handled this ino. After that, we are good...I think.
I'm not saying the code has be changed/optimized at this point, just
testing my understanding:)
>> struct btrfs_root_item:
>> what is the difference between existing "generation" field that you
>> mimic in generation_v2 and "ctransid"? (I know I need to study the
>> root tree code).
> Did you read the comments for btrfs_root_item and
> btrfs_read_root_item? They should answer your question.
Yes, I read the comments, but they address only "generation" and
"generation_v2", which I understand. Basically, I don't understand how
"generation" differs from ctransid (I need to dig more there).
Thanks!
Alex.
>>
>> Thank you,
>> Alex.
> Big thanks for your reviews :)
> I pushed a new version of the kernel and progs side. The branches are
> now "for-chris"
^ permalink raw reply [flat|nested] 3+ messages in thread
end of thread, other threads:[~2012-08-01 17:31 UTC | newest]
Thread overview: 3+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2012-07-31 16:32 Questions and notes about send/receive Alex Lyakas
2012-08-01 11:10 ` Alexander Block
2012-08-01 17:31 ` Alex Lyakas
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).