Update libglnx

Update libglnx, which is mostly port the repo stagedir code
to the new tmpdir API.  This turned out to require some
libglnx changes to support de-allocating the tmpdir ref while
still maintaining the on-disk dir.

Update submodule: libglnx

Closes: #1172
Approved by: jlebon
This commit is contained in:
Colin Walters 2017-09-12 09:40:58 -04:00 committed by Atomic Bot
parent b39a61b493
commit d0b0578cc1
6 changed files with 62 additions and 98 deletions

@ -1 +1 @@
Subproject commit e226ccf6913d1d852fde1e150a99fab508f85c34 Subproject commit e5856ca2939dca0589a836e3108dd3f9759e28fa

View File

@ -140,7 +140,7 @@ _ostree_repo_commit_tmpf_final (OstreeRepo *self,
int dest_dfd; int dest_dfd;
if (self->in_transaction) if (self->in_transaction)
dest_dfd = self->commit_stagedir_fd; dest_dfd = self->commit_stagedir.fd;
else else
dest_dfd = self->objects_dir_fd; dest_dfd = self->objects_dir_fd;
@ -173,7 +173,7 @@ _ostree_repo_commit_path_final (OstreeRepo *self,
int dest_dfd; int dest_dfd;
if (self->in_transaction) if (self->in_transaction)
dest_dfd = self->commit_stagedir_fd; dest_dfd = self->commit_stagedir.fd;
else else
dest_dfd = self->objects_dir_fd; dest_dfd = self->objects_dir_fd;
@ -1114,8 +1114,7 @@ ostree_repo_prepare_transaction (OstreeRepo *self,
gboolean ret_transaction_resume = FALSE; gboolean ret_transaction_resume = FALSE;
if (!_ostree_repo_allocate_tmpdir (self->tmp_dir_fd, if (!_ostree_repo_allocate_tmpdir (self->tmp_dir_fd,
self->stagedir_prefix, self->stagedir_prefix,
&self->commit_stagedir_name, &self->commit_stagedir,
&self->commit_stagedir_fd,
&self->commit_stagedir_lock, &self->commit_stagedir_lock,
&ret_transaction_resume, &ret_transaction_resume,
cancellable, error)) cancellable, error))
@ -1134,7 +1133,7 @@ rename_pending_loose_objects (OstreeRepo *self,
GLNX_AUTO_PREFIX_ERROR ("rename pending", error); GLNX_AUTO_PREFIX_ERROR ("rename pending", error);
g_auto(GLnxDirFdIterator) dfd_iter = { 0, }; g_auto(GLnxDirFdIterator) dfd_iter = { 0, };
if (!glnx_dirfd_iterator_init_at (self->commit_stagedir_fd, ".", FALSE, &dfd_iter, error)) if (!glnx_dirfd_iterator_init_at (self->commit_stagedir.fd, ".", FALSE, &dfd_iter, error))
return FALSE; return FALSE;
/* Iterate over the outer checksum dir */ /* Iterate over the outer checksum dir */
@ -1212,8 +1211,7 @@ rename_pending_loose_objects (OstreeRepo *self,
if (fsync (self->objects_dir_fd) == -1) if (fsync (self->objects_dir_fd) == -1)
return glnx_throw_errno_prefix (error, "fsync"); return glnx_throw_errno_prefix (error, "fsync");
if (!glnx_shutil_rm_rf_at (self->tmp_dir_fd, self->commit_stagedir_name, if (!glnx_tmpdir_delete (&self->commit_stagedir, cancellable, error))
cancellable, error))
return FALSE; return FALSE;
return TRUE; return TRUE;
@ -1549,15 +1547,8 @@ ostree_repo_commit_transaction (OstreeRepo *self,
return FALSE; return FALSE;
g_clear_pointer (&self->txn_collection_refs, g_hash_table_destroy); g_clear_pointer (&self->txn_collection_refs, g_hash_table_destroy);
if (self->commit_stagedir_fd != -1) glnx_tmpdir_unset (&self->commit_stagedir);
{ glnx_release_lock_file (&self->commit_stagedir_lock);
(void) close (self->commit_stagedir_fd);
self->commit_stagedir_fd = -1;
glnx_release_lock_file (&self->commit_stagedir_lock);
}
g_clear_pointer (&self->commit_stagedir_name, g_free);
self->in_transaction = FALSE; self->in_transaction = FALSE;
@ -1588,14 +1579,8 @@ ostree_repo_abort_transaction (OstreeRepo *self,
g_clear_pointer (&self->txn_refs, g_hash_table_destroy); g_clear_pointer (&self->txn_refs, g_hash_table_destroy);
g_clear_pointer (&self->txn_collection_refs, g_hash_table_destroy); g_clear_pointer (&self->txn_collection_refs, g_hash_table_destroy);
if (self->commit_stagedir_fd != -1) glnx_tmpdir_unset (&self->commit_stagedir);
{ glnx_release_lock_file (&self->commit_stagedir_lock);
(void) close (self->commit_stagedir_fd);
self->commit_stagedir_fd = -1;
glnx_release_lock_file (&self->commit_stagedir_lock);
}
g_clear_pointer (&self->commit_stagedir_name, g_free);
self->in_transaction = FALSE; self->in_transaction = FALSE;
@ -2181,8 +2166,8 @@ ostree_repo_read_commit_detached_metadata (OstreeRepo *self,
_ostree_loose_path (buf, checksum, OSTREE_OBJECT_TYPE_COMMIT_META, self->mode); _ostree_loose_path (buf, checksum, OSTREE_OBJECT_TYPE_COMMIT_META, self->mode);
g_autoptr(GVariant) ret_metadata = NULL; g_autoptr(GVariant) ret_metadata = NULL;
if (self->commit_stagedir_fd != -1 && if (self->commit_stagedir.initialized &&
!ot_util_variant_map_at (self->commit_stagedir_fd, buf, !ot_util_variant_map_at (self->commit_stagedir.fd, buf,
G_VARIANT_TYPE ("a{sv}"), G_VARIANT_TYPE ("a{sv}"),
OT_VARIANT_MAP_ALLOW_NOENT | OT_VARIANT_MAP_TRUSTED, &ret_metadata, error)) OT_VARIANT_MAP_ALLOW_NOENT | OT_VARIANT_MAP_TRUSTED, &ret_metadata, error))
return glnx_prefix_error (error, "Unable to read existing detached metadata"); return glnx_prefix_error (error, "Unable to read existing detached metadata");
@ -2229,7 +2214,7 @@ ostree_repo_write_commit_detached_metadata (OstreeRepo *self,
int dest_dfd; int dest_dfd;
if (self->in_transaction) if (self->in_transaction)
dest_dfd = self->commit_stagedir_fd; dest_dfd = self->commit_stagedir.fd;
else else
dest_dfd = self->objects_dir_fd; dest_dfd = self->objects_dir_fd;

View File

@ -92,8 +92,7 @@ struct OstreeRepo {
GObject parent; GObject parent;
char *stagedir_prefix; char *stagedir_prefix;
int commit_stagedir_fd; GLnxTmpDir commit_stagedir;
char *commit_stagedir_name;
GLnxLockFile commit_stagedir_lock; GLnxLockFile commit_stagedir_lock;
/* A cached fd-relative version, distinct from the case where we may have a /* A cached fd-relative version, distinct from the case where we may have a
@ -209,8 +208,7 @@ G_DEFINE_AUTO_CLEANUP_CLEAR_FUNC(OstreeRepoMemoryCacheRef, _ostree_repo_memory_c
gboolean gboolean
_ostree_repo_allocate_tmpdir (int tmpdir_dfd, _ostree_repo_allocate_tmpdir (int tmpdir_dfd,
const char *tmpdir_prefix, const char *tmpdir_prefix,
char **tmpdir_name_out, GLnxTmpDir *tmpdir_out,
int *tmpdir_fd_out,
GLnxLockFile *file_lock_out, GLnxLockFile *file_lock_out,
gboolean * reusing_dir_out, gboolean * reusing_dir_out,
GCancellable *cancellable, GCancellable *cancellable,

View File

@ -487,9 +487,7 @@ ostree_repo_finalize (GObject *object)
g_clear_object (&self->repodir); g_clear_object (&self->repodir);
if (self->repo_dir_fd != -1) if (self->repo_dir_fd != -1)
(void) close (self->repo_dir_fd); (void) close (self->repo_dir_fd);
if (self->commit_stagedir_fd != -1) glnx_tmpdir_unset (&self->commit_stagedir);
(void) close (self->commit_stagedir_fd);
g_free (self->commit_stagedir_name);
glnx_release_lock_file (&self->commit_stagedir_lock); glnx_release_lock_file (&self->commit_stagedir_lock);
if (self->tmp_dir_fd != -1) if (self->tmp_dir_fd != -1)
(void) close (self->tmp_dir_fd); (void) close (self->tmp_dir_fd);
@ -687,7 +685,6 @@ ostree_repo_init (OstreeRepo *self)
self->repo_dir_fd = -1; self->repo_dir_fd = -1;
self->cache_dir_fd = -1; self->cache_dir_fd = -1;
self->tmp_dir_fd = -1; self->tmp_dir_fd = -1;
self->commit_stagedir_fd = -1;
self->objects_dir_fd = -1; self->objects_dir_fd = -1;
self->uncompressed_objects_dir_fd = -1; self->uncompressed_objects_dir_fd = -1;
self->commit_stagedir_lock = empty_lockfile; self->commit_stagedir_lock = empty_lockfile;
@ -2793,9 +2790,9 @@ load_metadata_internal (OstreeRepo *self,
error)) error))
return FALSE; return FALSE;
if (fd < 0 && self->commit_stagedir_fd != -1) if (fd < 0 && self->commit_stagedir.initialized)
{ {
if (!ot_openat_ignore_enoent (self->commit_stagedir_fd, loose_path_buf, &fd, if (!ot_openat_ignore_enoent (self->commit_stagedir.fd, loose_path_buf, &fd,
error)) error))
return FALSE; return FALSE;
} }
@ -2906,9 +2903,9 @@ repo_load_file_archive (OstreeRepo *self,
error)) error))
return FALSE; return FALSE;
if (fd < 0 && self->commit_stagedir_fd != -1) if (fd < 0 && self->commit_stagedir.initialized)
{ {
if (!ot_openat_ignore_enoent (self->commit_stagedir_fd, loose_path_buf, &fd, if (!ot_openat_ignore_enoent (self->commit_stagedir.fd, loose_path_buf, &fd,
error)) error))
return FALSE; return FALSE;
} }
@ -2967,9 +2964,9 @@ _ostree_repo_load_file_bare (OstreeRepo *self,
int objdir_fd = self->objects_dir_fd; int objdir_fd = self->objects_dir_fd;
int res; int res;
if ((res = TEMP_FAILURE_RETRY (fstatat (objdir_fd, loose_path_buf, &stbuf, AT_SYMLINK_NOFOLLOW))) < 0 if ((res = TEMP_FAILURE_RETRY (fstatat (objdir_fd, loose_path_buf, &stbuf, AT_SYMLINK_NOFOLLOW))) < 0
&& errno == ENOENT && self->commit_stagedir_fd != -1) && errno == ENOENT && self->commit_stagedir.initialized)
{ {
objdir_fd = self->commit_stagedir_fd; objdir_fd = self->commit_stagedir.fd;
res = TEMP_FAILURE_RETRY (fstatat (objdir_fd, loose_path_buf, &stbuf, AT_SYMLINK_NOFOLLOW)); res = TEMP_FAILURE_RETRY (fstatat (objdir_fd, loose_path_buf, &stbuf, AT_SYMLINK_NOFOLLOW));
} }
if (res < 0 && errno != ENOENT) if (res < 0 && errno != ENOENT)
@ -3214,7 +3211,9 @@ _ostree_repo_has_loose_object (OstreeRepo *self,
gboolean found = FALSE; gboolean found = FALSE;
/* It's easier to share code if we make this an array */ /* It's easier to share code if we make this an array */
const int dfd_searches[] = { self->commit_stagedir_fd, self->objects_dir_fd }; int dfd_searches[] = { -1, self->objects_dir_fd };
if (self->commit_stagedir.initialized)
dfd_searches[0] = self->commit_stagedir.fd;
for (guint i = 0; i < G_N_ELEMENTS (dfd_searches); i++) for (guint i = 0; i < G_N_ELEMENTS (dfd_searches); i++)
{ {
int dfd = dfd_searches[i]; int dfd = dfd_searches[i];
@ -3654,8 +3653,8 @@ ostree_repo_query_object_storage_size (OstreeRepo *self,
struct stat stbuf; struct stat stbuf;
res = TEMP_FAILURE_RETRY (fstatat (self->objects_dir_fd, loose_path, &stbuf, AT_SYMLINK_NOFOLLOW)); res = TEMP_FAILURE_RETRY (fstatat (self->objects_dir_fd, loose_path, &stbuf, AT_SYMLINK_NOFOLLOW));
if (res < 0 && errno == ENOENT && self->commit_stagedir_fd != -1) if (res < 0 && errno == ENOENT && self->commit_stagedir.initialized)
res = TEMP_FAILURE_RETRY (fstatat (self->commit_stagedir_fd, loose_path, &stbuf, AT_SYMLINK_NOFOLLOW)); res = TEMP_FAILURE_RETRY (fstatat (self->commit_stagedir.fd, loose_path, &stbuf, AT_SYMLINK_NOFOLLOW));
if (res < 0) if (res < 0)
return glnx_throw_errno_prefix (error, "Querying object %s.%s", sha256, ostree_object_type_to_string (objtype)); return glnx_throw_errno_prefix (error, "Querying object %s.%s", sha256, ostree_object_type_to_string (objtype));
@ -5104,8 +5103,7 @@ _ostree_repo_try_lock_tmpdir (int tmpdir_dfd,
gboolean gboolean
_ostree_repo_allocate_tmpdir (int tmpdir_dfd, _ostree_repo_allocate_tmpdir (int tmpdir_dfd,
const char *tmpdir_prefix, const char *tmpdir_prefix,
char **tmpdir_name_out, GLnxTmpDir *tmpdir_out,
int *tmpdir_fd_out,
GLnxLockFile *file_lock_out, GLnxLockFile *file_lock_out,
gboolean *reusing_dir_out, gboolean *reusing_dir_out,
GCancellable *cancellable, GCancellable *cancellable,
@ -5120,9 +5118,8 @@ _ostree_repo_allocate_tmpdir (int tmpdir_dfd,
gboolean reusing_dir = FALSE; gboolean reusing_dir = FALSE;
gboolean did_lock = FALSE; gboolean did_lock = FALSE;
g_autofree char *tmpdir_name = NULL; g_auto(GLnxTmpDir) ret_tmpdir = { 0, };
glnx_fd_close int tmpdir_fd = -1; while (!ret_tmpdir.initialized)
while (tmpdir_name == NULL)
{ {
struct dirent *dent; struct dirent *dent;
glnx_fd_close int existing_tmpdir_fd = -1; glnx_fd_close int existing_tmpdir_fd = -1;
@ -5142,8 +5139,9 @@ _ostree_repo_allocate_tmpdir (int tmpdir_dfd,
dent->d_type != DT_DIR) dent->d_type != DT_DIR)
continue; continue;
glnx_fd_close int target_dfd = -1;
if (!glnx_opendirat (dfd_iter.fd, dent->d_name, FALSE, if (!glnx_opendirat (dfd_iter.fd, dent->d_name, FALSE,
&existing_tmpdir_fd, &local_error)) &target_dfd, &local_error))
{ {
if (g_error_matches (local_error, G_IO_ERROR, G_IO_ERROR_NOT_DIRECTORY)) if (g_error_matches (local_error, G_IO_ERROR, G_IO_ERROR_NOT_DIRECTORY))
continue; continue;
@ -5166,24 +5164,22 @@ _ostree_repo_allocate_tmpdir (int tmpdir_dfd,
/* Touch the reused directory so that we don't accidentally /* Touch the reused directory so that we don't accidentally
* remove it due to being old when cleaning up the tmpdir. * remove it due to being old when cleaning up the tmpdir.
*/ */
(void)futimens (existing_tmpdir_fd, NULL); (void)futimens (target_dfd, NULL);
/* We found an existing tmpdir which we managed to lock */ /* We found an existing tmpdir which we managed to lock */
tmpdir_name = g_strdup (dent->d_name); ret_tmpdir.src_dfd = tmpdir_dfd;
tmpdir_fd = glnx_steal_fd (&existing_tmpdir_fd); ret_tmpdir.fd = glnx_steal_fd (&target_dfd);
reusing_dir = TRUE; ret_tmpdir.path = g_strdup (dent->d_name);
ret_tmpdir.initialized = TRUE;
} }
while (tmpdir_name == NULL) while (!ret_tmpdir.initialized)
{ {
g_auto(GLnxTmpDir) new_tmpdir = { 0, };
/* No existing tmpdir found, create a new */ /* No existing tmpdir found, create a new */
g_autofree char *tmpdir_name_template = g_strconcat (tmpdir_prefix, "XXXXXX", NULL); const char *tmpdir_name_template = glnx_strjoina (tmpdir_prefix, "XXXXXX");
if (!glnx_mkdtempat (tmpdir_dfd, tmpdir_name_template, 0777, error)) if (!glnx_mkdtempat (tmpdir_dfd, tmpdir_name_template, 0755,
return FALSE; &new_tmpdir, error))
glnx_fd_close int new_tmpdir_fd = -1;
if (!glnx_opendirat (tmpdir_dfd, tmpdir_name_template, FALSE,
&new_tmpdir_fd, error))
return FALSE; return FALSE;
/* Note, at this point we can race with another process that picks up this /* Note, at this point we can race with another process that picks up this
@ -5195,16 +5191,13 @@ _ostree_repo_allocate_tmpdir (int tmpdir_dfd,
if (!did_lock) if (!did_lock)
continue; continue;
tmpdir_name = g_steal_pointer (&tmpdir_name_template); ret_tmpdir = new_tmpdir; /* Transfer ownership */
tmpdir_fd = glnx_steal_fd (&new_tmpdir_fd); new_tmpdir.initialized = FALSE;
} }
if (tmpdir_name_out) *tmpdir_out = ret_tmpdir; /* Transfer ownership */
*tmpdir_name_out = g_steal_pointer (&tmpdir_name); ret_tmpdir.initialized = FALSE;
if (tmpdir_fd_out) *reusing_dir_out = reusing_dir;
*tmpdir_fd_out = glnx_steal_fd (&tmpdir_fd);
if (reusing_dir_out)
*reusing_dir_out = reusing_dir;
return TRUE; return TRUE;
} }

View File

@ -39,7 +39,7 @@
typedef struct typedef struct
{ {
OstreeRepo *parent_repo; /* owned */ OstreeRepo *parent_repo; /* owned */
int working_dfd; /* owned */ GLnxTmpDir tmpdir; /* owned */
GFile *working_dir; /* owned */ GFile *working_dir; /* owned */
} Fixture; } Fixture;
@ -47,20 +47,17 @@ static void
setup (Fixture *fixture, setup (Fixture *fixture,
gconstpointer test_data) gconstpointer test_data)
{ {
g_autofree gchar *tmp_name = NULL;
g_autoptr(GError) error = NULL; g_autoptr(GError) error = NULL;
tmp_name = g_strdup ("test-repo-finder-config-XXXXXX"); (void)glnx_mkdtemp ("test-repo-finder-config-XXXXXX", 0700, &fixture->tmpdir, &error);
glnx_mkdtempat_open_in_system (tmp_name, 0700, &fixture->working_dfd, &error);
g_assert_no_error (error); g_assert_no_error (error);
g_test_message ("Using temporary directory: %s", tmp_name); g_test_message ("Using temporary directory: %s", fixture->tmpdir.path);
glnx_shutil_mkdir_p_at (fixture->working_dfd, "repo", 0700, NULL, &error); glnx_shutil_mkdir_p_at (fixture->tmpdir.fd, "repo", 0700, NULL, &error);
g_assert_no_error (error); g_assert_no_error (error);
g_autoptr(GFile) tmp_dir = g_file_new_for_path (g_get_tmp_dir ()); fixture->working_dir = g_file_new_for_path (fixture->tmpdir.path);
fixture->working_dir = g_file_get_child (tmp_dir, tmp_name);
fixture->parent_repo = ot_test_setup_repo (NULL, &error); fixture->parent_repo = ot_test_setup_repo (NULL, &error);
g_assert_no_error (error); g_assert_no_error (error);
@ -73,10 +70,7 @@ teardown (Fixture *fixture,
g_autoptr(GError) error = NULL; g_autoptr(GError) error = NULL;
/* Recursively remove the temporary directory. */ /* Recursively remove the temporary directory. */
glnx_shutil_rm_rf_at (fixture->working_dfd, ".", NULL, NULL); (void)glnx_tmpdir_delete (&fixture->tmpdir, NULL, NULL);
close (fixture->working_dfd);
fixture->working_dfd = -1;
/* The repo also needs its source files to be removed. This is the inverse /* The repo also needs its source files to be removed. This is the inverse
* of setup_test_repository() in libtest.sh. */ * of setup_test_repository() in libtest.sh. */
@ -177,7 +171,7 @@ assert_create_remote (Fixture *fixture,
g_autoptr(GError) error = NULL; g_autoptr(GError) error = NULL;
const gchar *repo_name = (collection_id != NULL) ? collection_id : "no-collection"; const gchar *repo_name = (collection_id != NULL) ? collection_id : "no-collection";
glnx_shutil_mkdir_p_at (fixture->working_dfd, repo_name, 0700, NULL, &error); glnx_shutil_mkdir_p_at (fixture->tmpdir.fd, repo_name, 0700, NULL, &error);
g_assert_no_error (error); g_assert_no_error (error);
g_autoptr(GFile) repo_path = g_file_get_child (fixture->working_dir, repo_name); g_autoptr(GFile) repo_path = g_file_get_child (fixture->working_dir, repo_name);

View File

@ -40,30 +40,27 @@
typedef struct typedef struct
{ {
OstreeRepo *parent_repo; OstreeRepo *parent_repo;
int working_dfd; /* owned */ GLnxTmpDir tmpdir; /* owned */
GFile *working_dir; /* owned */ GFile *working_dir; /* Points at tmpdir */
} Fixture; } Fixture;
static void static void
setup (Fixture *fixture, setup (Fixture *fixture,
gconstpointer test_data) gconstpointer test_data)
{ {
g_autofree gchar *tmp_name = NULL;
g_autoptr(GError) error = NULL; g_autoptr(GError) error = NULL;
tmp_name = g_strdup ("test-repo-finder-mount-XXXXXX"); (void)glnx_mkdtemp ("test-repo-finder-mount-XXXXXX", 0700, &fixture->tmpdir, &error);
glnx_mkdtempat_open_in_system (tmp_name, 0700, &fixture->working_dfd, &error);
g_assert_no_error (error); g_assert_no_error (error);
g_test_message ("Using temporary directory: %s", tmp_name); g_test_message ("Using temporary directory: %s", fixture->tmpdir.path);
glnx_shutil_mkdir_p_at (fixture->working_dfd, "repo", 0700, NULL, &error); glnx_shutil_mkdir_p_at (fixture->tmpdir.fd, "repo", 0700, NULL, &error);
if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_EXISTS)) if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_EXISTS))
g_clear_error (&error); g_clear_error (&error);
g_assert_no_error (error); g_assert_no_error (error);
g_autoptr(GFile) tmp_dir = g_file_new_for_path (g_get_tmp_dir ()); fixture->working_dir = g_file_new_for_path (fixture->tmpdir.path);
fixture->working_dir = g_file_get_child (tmp_dir, tmp_name);
fixture->parent_repo = ot_test_setup_repo (NULL, &error); fixture->parent_repo = ot_test_setup_repo (NULL, &error);
g_assert_no_error (error); g_assert_no_error (error);
@ -76,10 +73,7 @@ teardown (Fixture *fixture,
g_autoptr(GError) error = NULL; g_autoptr(GError) error = NULL;
/* Recursively remove the temporary directory. */ /* Recursively remove the temporary directory. */
glnx_shutil_rm_rf_at (fixture->working_dfd, ".", NULL, NULL); (void)glnx_tmpdir_delete (&fixture->tmpdir, NULL, NULL);
close (fixture->working_dfd);
fixture->working_dfd = -1;
/* The repo also needs its source files to be removed. This is the inverse /* The repo also needs its source files to be removed. This is the inverse
* of setup_test_repository() in libtest.sh. */ * of setup_test_repository() in libtest.sh. */
@ -167,7 +161,7 @@ assert_create_repos_dir (Fixture *fixture,
g_autoptr(GError) error = NULL; g_autoptr(GError) error = NULL;
g_autofree gchar *path = g_build_filename (mount_root_name, ".ostree", "repos", NULL); g_autofree gchar *path = g_build_filename (mount_root_name, ".ostree", "repos", NULL);
glnx_shutil_mkdir_p_at_open (fixture->working_dfd, path, 0700, &repos_dfd, NULL, &error); glnx_shutil_mkdir_p_at_open (fixture->tmpdir.fd, path, 0700, &repos_dfd, NULL, &error);
if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_EXISTS)) if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_EXISTS))
g_clear_error (&error); g_clear_error (&error);
g_assert_no_error (error); g_assert_no_error (error);