Skip to content

Commit

Permalink
sysroot-deploy: Fix full_system_sync
Browse files Browse the repository at this point in the history
  • Loading branch information
ruihe774 committed Dec 23, 2024
1 parent cc10e37 commit 3623ec0
Show file tree
Hide file tree
Showing 3 changed files with 66 additions and 50 deletions.
37 changes: 22 additions & 15 deletions src/libostree/ostree-sysroot-deploy.c
Original file line number Diff line number Diff line change
Expand Up @@ -1651,6 +1651,18 @@ full_system_sync (OstreeSysroot *self, SyncStats *out_stats, GCancellable *cance
GError **error)
{
GLNX_AUTO_PREFIX_ERROR ("Full sync", error);

gboolean in_root;
if (!_ostree_sysroot_enter_mount_namespace (self, &in_root, error))
return FALSE;

glnx_autofd int old_sysroot_fd = g_steal_fd (&self->sysroot_fd);
glnx_autofd int old_boot_fd = g_steal_fd (&self->boot_fd);
if (!_ostree_sysroot_ensure_sysroot_fd (self, error))
return FALSE;
if (!_ostree_sysroot_ensure_boot_fd (self, error))
return FALSE;

ot_journal_print (LOG_INFO, "Starting syncfs() for system root");
guint64 start_msec = g_get_monotonic_time () / 1000;
if (syncfs (self->sysroot_fd) != 0)
Expand All @@ -1675,6 +1687,13 @@ full_system_sync (OstreeSysroot *self, SyncStats *out_stats, GCancellable *cance
end_msec - start_msec);
out_stats->boot_syncfs_msec = (end_msec - start_msec);

ostree_sysroot_unload (self);
self->sysroot_fd = g_steal_fd (&old_sysroot_fd);
self->boot_fd = g_steal_fd (&old_boot_fd);

if (in_root && !_ostree_enter_root_mount_namespace (error))
return FALSE;

return TRUE;
}

Expand Down Expand Up @@ -3340,10 +3359,7 @@ _ostree_sysroot_run_in_deployment (OstreeSysroot *sysroot, int deployment_dfd,
gboolean success = FALSE;

gboolean in_root;
if (!_ostree_in_root_mount_namespace (&in_root, error))
return FALSE;

if (!_ostree_sysroot_enter_mount_namespace (sysroot, error))
if (!_ostree_sysroot_enter_mount_namespace (sysroot, &in_root, error))
return FALSE;

g_autoptr (GPtrArray) args = g_ptr_array_new ();
Expand Down Expand Up @@ -3383,17 +3399,8 @@ _ostree_sysroot_run_in_deployment (OstreeSysroot *sysroot, int deployment_dfd,
success = g_spawn_sync (NULL, (char **)args->pdata, NULL, G_SPAWN_LEAVE_DESCRIPTORS_OPEN, NULL,
(gpointer)(uintptr_t)deployment_dfd, stdout, NULL, exit_status, error);

/* Switch back */
if (in_root)
{
glnx_autofd int root_ns_fd = -1;

if (!glnx_openat_rdonly (AT_FDCWD, "/proc/1/ns/mnt", TRUE, &root_ns_fd, error))
return FALSE;

if (setns (root_ns_fd, CLONE_NEWNS) < 0)
return glnx_throw_errno_prefix (error, "setns");
}
if (in_root && !_ostree_enter_root_mount_namespace (error))
return FALSE;

fail:
for (int *it = bind_fds; it && *it >= 0; it++)
Expand Down
6 changes: 5 additions & 1 deletion src/libostree/ostree-sysroot-private.h
Original file line number Diff line number Diff line change
Expand Up @@ -113,9 +113,13 @@ struct OstreeSysroot
// Relative to /boot, consumed by ostree-boot-complete.service
#define _OSTREE_FINALIZE_STAGED_FAILURE_PATH "ostree/finalize-failure.stamp"

gboolean _ostree_sysroot_ensure_sysroot_fd (OstreeSysroot *self, GError **error);

gboolean _ostree_in_root_mount_namespace (gboolean *out_val, GError **error);

gboolean _ostree_sysroot_enter_mount_namespace (OstreeSysroot *self, GError **error);
gboolean _ostree_sysroot_enter_mount_namespace (OstreeSysroot *self, gboolean *out_in_root, GError **error);

gboolean _ostree_enter_root_mount_namespace (GError **error);

gboolean _ostree_sysroot_ensure_writable (OstreeSysroot *self, GError **error);

Expand Down
73 changes: 39 additions & 34 deletions src/libostree/ostree-sysroot.c
Original file line number Diff line number Diff line change
Expand Up @@ -277,8 +277,8 @@ ostree_sysroot_set_mount_namespace_in_use (OstreeSysroot *self)
g_assert (_ostree_in_root_mount_namespace (&in_root, &local_error) && !in_root);
}

static gboolean
ensure_sysroot_fd (OstreeSysroot *self, GError **error);
gboolean
_ostree_sysroot_ensure_sysroot_fd (OstreeSysroot *self, GError **error);

gboolean
_ostree_sysroot_ensure_boot_fd (OstreeSysroot *self, GError **error);
Expand All @@ -287,14 +287,17 @@ static gboolean
_ostree_sysroot_invisible (const OstreeSysroot *self, gboolean *out_val, GError **error);

gboolean
_ostree_sysroot_enter_mount_namespace (OstreeSysroot *self, GError **error)
_ostree_sysroot_enter_mount_namespace (OstreeSysroot *self, gboolean *out_in_root, GError **error)
{
gboolean in_root_stub;
if (!out_in_root)
out_in_root = &in_root_stub;

/* We also assume operating on non-booted roots won't have a readonly sysroot */
if (!self->root_is_ostree_booted)
return TRUE;

gboolean in_root;
if (!_ostree_in_root_mount_namespace (&in_root, error))
if (!_ostree_in_root_mount_namespace (out_in_root, error))
return FALSE;

/* Backup tree fd of sysroot_fd and boot_fd */
Expand Down Expand Up @@ -326,7 +329,7 @@ _ostree_sysroot_enter_mount_namespace (OstreeSysroot *self, GError **error)
}

// If the mount namespaces are the same, we need to unshare().
if (in_root)
if (*out_in_root)
{
if (unshare (CLONE_NEWNS) < 0)
return glnx_throw_errno_prefix (error, "Failed to invoke unshare(CLONE_NEWNS)");
Expand All @@ -338,7 +341,7 @@ _ostree_sysroot_enter_mount_namespace (OstreeSysroot *self, GError **error)

/* Mount sysroot and boot back */
ostree_sysroot_unload (self);
if (!ensure_sysroot_fd (self, error))
if (!_ostree_sysroot_ensure_sysroot_fd (self, error))
return FALSE;

gboolean invisible;
Expand Down Expand Up @@ -376,6 +379,20 @@ _ostree_sysroot_enter_mount_namespace (OstreeSysroot *self, GError **error)
return TRUE;
}

gboolean
_ostree_enter_root_mount_namespace (GError **error)
{
glnx_autofd int root_ns_fd = -1;

if (!glnx_openat_rdonly (AT_FDCWD, "/proc/1/ns/mnt", TRUE, &root_ns_fd, error))
return FALSE;

if (setns (root_ns_fd, CLONE_NEWNS) < 0)
return glnx_throw_errno_prefix (error, "setns");

return TRUE;
}

/**
* ostree_sysroot_initialize_with_mount_namespace:
*
Expand Down Expand Up @@ -406,7 +423,7 @@ ostree_sysroot_initialize_with_mount_namespace (OstreeSysroot *self, GCancellabl
if (!ostree_sysroot_initialize (self, error))
return FALSE;

return _ostree_sysroot_enter_mount_namespace (self, error);
return _ostree_sysroot_enter_mount_namespace (self, NULL, error);
}

/**
Expand All @@ -422,8 +439,8 @@ ostree_sysroot_get_path (OstreeSysroot *self)
}

/* Open a directory file descriptor for the sysroot if we haven't yet */
static gboolean
ensure_sysroot_fd (OstreeSysroot *self, GError **error)
gboolean
_ostree_sysroot_ensure_sysroot_fd (OstreeSysroot *self, GError **error)
{
if (self->sysroot_fd == -1)
{
Expand Down Expand Up @@ -566,10 +583,7 @@ _ostree_sysroot_ensure_writable (OstreeSysroot *self, GError **error)
return FALSE;

gboolean in_root;
if (!_ostree_in_root_mount_namespace (&in_root, error))
return FALSE;

if (!_ostree_sysroot_enter_mount_namespace (self, error))
if (!_ostree_sysroot_enter_mount_namespace (self, &in_root, error))
return FALSE;

ostree_sysroot_unload (self);
Expand All @@ -584,25 +598,16 @@ _ostree_sysroot_ensure_writable (OstreeSysroot *self, GError **error)
if (!remount_writable (boot_path, &did_remount_boot, error))
return FALSE;

if (!ensure_sysroot_fd (self, error))
if (!_ostree_sysroot_ensure_sysroot_fd (self, error))
return FALSE;
glnx_autofd int old_sysroot_fd = g_steal_fd (&self->sysroot_fd);
if ((self->sysroot_fd = (int)syscall (SYS_open_tree, old_sysroot_fd, "sysroot", 1 /* OPEN_TREE_CLONE */ | O_CLOEXEC)) < 0)
return glnx_throw_errno_prefix (error, "open_tree");
if ((self->boot_fd = (int)syscall (SYS_open_tree, old_sysroot_fd, "boot", 1 /* OPEN_TREE_CLONE */ | O_CLOEXEC)) < 0)
return glnx_throw_errno_prefix (error, "open_tree");

/* Switch back */
if (in_root)
{
glnx_autofd int root_ns_fd = -1;

if (!glnx_openat_rdonly (AT_FDCWD, "/proc/1/ns/mnt", TRUE, &root_ns_fd, error))
return FALSE;

if (setns (root_ns_fd, CLONE_NEWNS) < 0)
return glnx_throw_errno_prefix (error, "setns");
}
if (in_root && !_ostree_enter_root_mount_namespace (error))
return FALSE;

return TRUE;
}
Expand Down Expand Up @@ -682,7 +687,7 @@ ostree_sysroot_unload (OstreeSysroot *self)
gboolean
ostree_sysroot_ensure_initialized (OstreeSysroot *self, GCancellable *cancellable, GError **error)
{
if (!ensure_sysroot_fd (self, error))
if (!_ostree_sysroot_ensure_sysroot_fd (self, error))
return FALSE;

if (!glnx_shutil_mkdir_p_at (self->sysroot_fd, "ostree/repo", 0755, cancellable, error))
Expand Down Expand Up @@ -736,7 +741,7 @@ _ostree_sysroot_read_current_subbootversion (OstreeSysroot *self, int bootversio
{
GLNX_AUTO_PREFIX_ERROR ("Reading current subbootversion", error);

if (!ensure_sysroot_fd (self, error))
if (!_ostree_sysroot_ensure_sysroot_fd (self, error))
return FALSE;

g_autofree char *ostree_bootdir_name = g_strdup_printf ("ostree/boot.%d", bootversion);
Expand Down Expand Up @@ -802,7 +807,7 @@ _ostree_sysroot_read_boot_loader_configs (OstreeSysroot *self, int bootversion,
GPtrArray **out_loader_configs, GCancellable *cancellable,
GError **error)
{
if (!ensure_sysroot_fd (self, error))
if (!_ostree_sysroot_ensure_sysroot_fd (self, error))
return FALSE;

g_autoptr (GPtrArray) ret_loader_configs
Expand Down Expand Up @@ -963,7 +968,7 @@ static gboolean
parse_deployment (OstreeSysroot *self, const char *boot_link, OstreeDeployment **out_deployment,
GCancellable *cancellable, GError **error)
{
if (!ensure_sysroot_fd (self, error))
if (!_ostree_sysroot_ensure_sysroot_fd (self, error))
return FALSE;

int entry_boot_version;
Expand Down Expand Up @@ -1177,7 +1182,7 @@ ensure_repo (OstreeSysroot *self, GError **error)

if (self->repo != NULL)
return TRUE;
if (!ensure_sysroot_fd (self, error))
if (!_ostree_sysroot_ensure_sysroot_fd (self, error))
return FALSE;
self->repo = ostree_repo_open_at (self->sysroot_fd, "ostree/repo", NULL, error);
if (!self->repo)
Expand Down Expand Up @@ -1215,7 +1220,7 @@ ostree_sysroot_initialize (OstreeSysroot *self, GError **error)
{
GLNX_AUTO_PREFIX_ERROR ("Initializing sysroot", error);

if (!ensure_sysroot_fd (self, error))
if (!_ostree_sysroot_ensure_sysroot_fd (self, error))
return FALSE;

if (self->loadstate < OSTREE_SYSROOT_LOAD_STATE_INIT)
Expand Down Expand Up @@ -1877,7 +1882,7 @@ ostree_sysroot_origin_new_from_refspec (OstreeSysroot *self, const char *refspec
gboolean
ostree_sysroot_lock (OstreeSysroot *self, GError **error)
{
if (!ensure_sysroot_fd (self, error))
if (!_ostree_sysroot_ensure_sysroot_fd (self, error))
return FALSE;

if (!_ostree_sysroot_ensure_writable (self, error))
Expand All @@ -1904,7 +1909,7 @@ ostree_sysroot_lock (OstreeSysroot *self, GError **error)
gboolean
ostree_sysroot_try_lock (OstreeSysroot *self, gboolean *out_acquired, GError **error)
{
if (!ensure_sysroot_fd (self, error))
if (!_ostree_sysroot_ensure_sysroot_fd (self, error))
return FALSE;

if (!_ostree_sysroot_ensure_writable (self, error))
Expand Down

0 comments on commit 3623ec0

Please sign in to comment.