-
Notifications
You must be signed in to change notification settings - Fork 305
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Rework devino caching to support invalidation if metadata changes #1170
Changes from 4 commits
7a76821
e73f709
87a2947
37a2fad
3433694
5cc1dc7
d1fa17c
9410a8f
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -2363,58 +2363,68 @@ ptrarray_path_join (GPtrArray *path) | |
} | ||
|
||
static gboolean | ||
get_modified_xattrs (OstreeRepo *self, | ||
OstreeRepoCommitModifier *modifier, | ||
const char *relpath, | ||
GFileInfo *file_info, | ||
GFile *path, | ||
int dfd, | ||
const char *dfd_subpath, | ||
GVariant **out_xattrs, | ||
GCancellable *cancellable, | ||
GError **error) | ||
{ | ||
g_autoptr(GVariant) ret_xattrs = NULL; | ||
|
||
if (modifier && modifier->xattr_callback) | ||
{ | ||
ret_xattrs = modifier->xattr_callback (self, relpath, file_info, | ||
modifier->xattr_user_data); | ||
} | ||
else if (!(modifier && (modifier->flags & (OSTREE_REPO_COMMIT_MODIFIER_FLAGS_SKIP_XATTRS | | ||
OSTREE_REPO_COMMIT_MODIFIER_FLAGS_CANONICAL_PERMISSIONS)) > 0) | ||
&& !self->disable_xattrs) | ||
get_final_xattrs (OstreeRepo *self, | ||
OstreeRepoCommitModifier *modifier, | ||
const char *relpath, | ||
GFileInfo *file_info, | ||
GFile *path, | ||
int dfd, | ||
const char *dfd_subpath, | ||
GVariant **out_xattrs, | ||
gboolean *out_modified, | ||
GCancellable *cancellable, | ||
GError **error) | ||
{ | ||
/* track whether the returned xattrs differ from the file on disk */ | ||
gboolean modified = TRUE; | ||
const gboolean skip_xattrs = (modifier && | ||
modifier->flags & (OSTREE_REPO_COMMIT_MODIFIER_FLAGS_SKIP_XATTRS | | ||
OSTREE_REPO_COMMIT_MODIFIER_FLAGS_CANONICAL_PERMISSIONS)) > 0; | ||
|
||
/* fetch on-disk xattrs if needed & not disabled */ | ||
g_autoptr(GVariant) original_xattrs = NULL; | ||
if (!skip_xattrs && !self->disable_xattrs) | ||
{ | ||
if (path && OSTREE_IS_REPO_FILE (path)) | ||
{ | ||
if (!ostree_repo_file_get_xattrs (OSTREE_REPO_FILE (path), | ||
&ret_xattrs, | ||
cancellable, | ||
error)) | ||
if (!ostree_repo_file_get_xattrs (OSTREE_REPO_FILE (path), &original_xattrs, | ||
cancellable, error)) | ||
return FALSE; | ||
} | ||
else if (path) | ||
{ | ||
if (!glnx_dfd_name_get_all_xattrs (AT_FDCWD, gs_file_get_path_cached (path), | ||
&ret_xattrs, cancellable, error)) | ||
&original_xattrs, cancellable, error)) | ||
return FALSE; | ||
} | ||
else if (dfd_subpath == NULL) | ||
{ | ||
g_assert (dfd != -1); | ||
if (!glnx_fd_get_all_xattrs (dfd, &ret_xattrs, | ||
cancellable, error)) | ||
if (!glnx_fd_get_all_xattrs (dfd, &original_xattrs, cancellable, error)) | ||
return FALSE; | ||
} | ||
else | ||
{ | ||
g_assert (dfd != -1); | ||
if (!glnx_dfd_name_get_all_xattrs (dfd, dfd_subpath, &ret_xattrs, | ||
cancellable, error)) | ||
if (!glnx_dfd_name_get_all_xattrs (dfd, dfd_subpath, &original_xattrs, | ||
cancellable, error)) | ||
return FALSE; | ||
} | ||
|
||
g_assert (original_xattrs); | ||
} | ||
|
||
g_autoptr(GVariant) ret_xattrs = NULL; | ||
if (modifier && modifier->xattr_callback) | ||
{ | ||
ret_xattrs = modifier->xattr_callback (self, relpath, file_info, | ||
modifier->xattr_user_data); | ||
} | ||
|
||
/* if callback returned NULL or didn't exist, default to on-disk state */ | ||
if (!ret_xattrs && original_xattrs) | ||
ret_xattrs = g_variant_ref (original_xattrs); | ||
|
||
if (modifier && modifier->sepolicy) | ||
{ | ||
g_autofree char *label = NULL; | ||
|
@@ -2436,10 +2446,9 @@ get_modified_xattrs (OstreeRepo *self, | |
{ | ||
/* drop out any existing SELinux policy from the set, so we don't end up | ||
* counting it twice in the checksum */ | ||
g_autoptr(GVariant) new_ret_xattrs = NULL; | ||
new_ret_xattrs = _ostree_filter_selinux_xattr (ret_xattrs); | ||
GVariant* new_ret_xattrs = _ostree_filter_selinux_xattr (ret_xattrs); | ||
g_variant_unref (ret_xattrs); | ||
ret_xattrs = g_steal_pointer (&new_ret_xattrs); | ||
ret_xattrs = new_ret_xattrs; | ||
} | ||
|
||
/* ret_xattrs may be NULL */ | ||
|
@@ -2458,8 +2467,13 @@ get_modified_xattrs (OstreeRepo *self, | |
} | ||
} | ||
|
||
if (original_xattrs && ret_xattrs && g_variant_equal (original_xattrs, ret_xattrs)) | ||
modified = FALSE; | ||
|
||
if (out_xattrs) | ||
*out_xattrs = g_steal_pointer (&ret_xattrs); | ||
if (out_modified) | ||
*out_modified = modified; | ||
return TRUE; | ||
} | ||
|
||
|
@@ -2506,6 +2520,7 @@ write_directory_content_to_mtree_internal (OstreeRepo *self, | |
g_autoptr(GFileInfo) modified_info = NULL; | ||
OstreeRepoCommitFilterResult filter_result = | ||
_ostree_repo_commit_modifier_apply (self, modifier, child_relpath, child_info, &modified_info); | ||
const gboolean child_info_was_modified = !_ostree_gfileinfo_equal (child_info, modified_info); | ||
|
||
if (filter_result != OSTREE_REPO_COMMIT_FILTER_ALLOW) | ||
{ | ||
|
@@ -2567,16 +2582,26 @@ write_directory_content_to_mtree_internal (OstreeRepo *self, | |
else | ||
{ | ||
guint64 file_obj_length; | ||
const char *loose_checksum; | ||
g_autoptr(GInputStream) file_input = NULL; | ||
g_autoptr(GVariant) xattrs = NULL; | ||
g_autoptr(GInputStream) file_object_input = NULL; | ||
g_autofree guchar *child_file_csum = NULL; | ||
g_autofree char *tmp_checksum = NULL; | ||
|
||
loose_checksum = devino_cache_lookup (self, modifier, | ||
g_file_info_get_attribute_uint32 (child_info, "unix::device"), | ||
g_file_info_get_attribute_uint64 (child_info, "unix::inode")); | ||
g_autoptr(GVariant) xattrs = NULL; | ||
gboolean xattrs_were_modified; | ||
if (!get_final_xattrs (self, modifier, child_relpath, child_info, child, | ||
dfd_iter != NULL ? dfd_iter->fd : -1, name, &xattrs, | ||
&xattrs_were_modified, cancellable, error)) | ||
return FALSE; | ||
|
||
/* only check the devino cache if the file info & xattrs were not modified */ | ||
const char *loose_checksum = NULL; | ||
if (!child_info_was_modified && !xattrs_were_modified) | ||
{ | ||
guint32 dev = g_file_info_get_attribute_uint32 (child_info, "unix::device"); | ||
guint64 inode = g_file_info_get_attribute_uint64 (child_info, "unix::inode"); | ||
loose_checksum = devino_cache_lookup (self, modifier, dev, inode); | ||
} | ||
|
||
if (loose_checksum) | ||
{ | ||
|
@@ -2602,12 +2627,6 @@ write_directory_content_to_mtree_internal (OstreeRepo *self, | |
} | ||
} | ||
|
||
if (!get_modified_xattrs (self, modifier, | ||
child_relpath, child_info, child, dfd_iter != NULL ? dfd_iter->fd : -1, name, | ||
&xattrs, | ||
cancellable, error)) | ||
return FALSE; | ||
|
||
if (!ostree_raw_file_to_content_stream (file_input, | ||
modified_info, xattrs, | ||
&file_object_input, &file_obj_length, | ||
|
@@ -2681,10 +2700,8 @@ write_directory_to_mtree_internal (OstreeRepo *self, | |
|
||
if (filter_result == OSTREE_REPO_COMMIT_FILTER_ALLOW) | ||
{ | ||
if (!get_modified_xattrs (self, modifier, relpath, child_info, | ||
dir, -1, NULL, | ||
&xattrs, | ||
cancellable, error)) | ||
if (!get_final_xattrs (self, modifier, relpath, child_info, dir, -1, NULL, | ||
&xattrs, NULL, cancellable, error)) | ||
return FALSE; | ||
|
||
g_autofree guchar *child_file_csum = NULL; | ||
|
@@ -2768,10 +2785,8 @@ write_dfd_iter_to_mtree_internal (OstreeRepo *self, | |
|
||
if (filter_result == OSTREE_REPO_COMMIT_FILTER_ALLOW) | ||
{ | ||
if (!get_modified_xattrs (self, modifier, relpath, modified_info, | ||
NULL, src_dfd_iter->fd, NULL, | ||
&xattrs, | ||
cancellable, error)) | ||
if (!get_final_xattrs (self, modifier, relpath, modified_info, NULL, src_dfd_iter->fd, | ||
NULL, &xattrs, NULL, cancellable, error)) | ||
return FALSE; | ||
|
||
if (!_ostree_repo_write_directory_meta (self, modified_info, xattrs, &child_file_csum, | ||
|
@@ -2801,16 +2816,6 @@ write_dfd_iter_to_mtree_internal (OstreeRepo *self, | |
if (!glnx_fstatat (src_dfd_iter->fd, dent->d_name, &stbuf, AT_SYMLINK_NOFOLLOW, error)) | ||
return FALSE; | ||
|
||
const char *loose_checksum = devino_cache_lookup (self, modifier, stbuf.st_dev, stbuf.st_ino); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think this patch should be included with the last commit since they're not |
||
if (loose_checksum) | ||
{ | ||
if (!ostree_mutable_tree_replace_file (mtree, dent->d_name, loose_checksum, | ||
error)) | ||
return FALSE; | ||
|
||
continue; | ||
} | ||
|
||
g_autoptr(GFileInfo) child_info = _ostree_stbuf_to_gfileinfo (&stbuf); | ||
g_file_info_set_name (child_info, dent->d_name); | ||
|
||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -19,7 +19,7 @@ | |
|
||
set -euo pipefail | ||
|
||
echo "1..$((72 + ${extra_basic_tests:-0}))" | ||
echo "1..$((73 + ${extra_basic_tests:-0}))" | ||
|
||
CHECKOUT_U_ARG="" | ||
CHECKOUT_H_ARGS="-H" | ||
|
@@ -602,6 +602,24 @@ $OSTREE checkout test2 test2-checkout | |
(cd test2-checkout && $OSTREE commit ${COMMIT_ARGS} --link-checkout-speedup -b test2 -s "tmp") | ||
echo "ok commit with link speedup" | ||
|
||
cd ${test_tmpdir} | ||
rm -rf test2-checkout | ||
$OSTREE checkout test2 test2-checkout | ||
# set cow to different perms, but re-set cowro to the same perms | ||
cat > statoverride.txt <<EOF | ||
=$((0600)) /baz/cow | ||
=$((0600)) /baz/cowro | ||
EOF | ||
$OSTREE commit ${COMMIT_ARGS} --statoverride=statoverride.txt \ | ||
--table-output --link-checkout-speedup -b test2-tmp test2-checkout > stats.txt | ||
$OSTREE diff test2 test2-tmp > diff-test2 | ||
assert_file_has_content diff-test2 'M */baz/cow$' | ||
assert_not_file_has_content diff-test2 'M */baz/cowro$' | ||
assert_not_file_has_content diff-test2 'baz/saucer' | ||
# only /baz/cow is a cache miss | ||
assert_file_has_content stats.txt '^Content Written: 1$' | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Ah, right I forgot we already had a way to unit test this. Cool. |
||
echo "ok commit with link speedup and modifier" | ||
|
||
cd ${test_tmpdir} | ||
$OSTREE ls test2 | ||
echo "ok ls with no argument" | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ooh, tricky.