repo/private: move OstreeRepoAutoTransaction to a boxed type

This defines `OstreeRepoAutoTransaction` as a boxed type, in order
to support auto-generating bindings for it.
That first requires adding internal reference-counting to it, to
allow freely copying/freeing references to a single transaction guard.
This commit is contained in:
Luca BRUNO 2021-10-11 06:52:25 +00:00
parent f4a3837662
commit 8a9737aa6e
No known key found for this signature in database
GPG Key ID: A9834A2252078E4E
2 changed files with 52 additions and 17 deletions

View File

@ -525,6 +525,7 @@ _ostree_repo_verify_bindings (const char *collection_id,
*/ */
typedef struct typedef struct
{ {
gint atomic_refcount;
OstreeRepo *repo; OstreeRepo *repo;
} OstreeRepoAutoTransaction; } OstreeRepoAutoTransaction;
@ -544,9 +545,14 @@ _ostree_repo_auto_transaction_commit (OstreeRepoAutoTransaction *txn,
GCancellable *cancellable, GCancellable *cancellable,
GError **error); GError **error);
void OstreeRepoAutoTransaction *
_ostree_repo_auto_transaction_cleanup (void *p); _ostree_repo_auto_transaction_ref (OstreeRepoAutoTransaction *txn);
G_DEFINE_AUTOPTR_CLEANUP_FUNC (OstreeRepoAutoTransaction, _ostree_repo_auto_transaction_cleanup); void
_ostree_repo_auto_transaction_unref (OstreeRepoAutoTransaction *txn);
GType _ostree_repo_auto_transaction_get_type (void);
G_DEFINE_AUTOPTR_CLEANUP_FUNC (OstreeRepoAutoTransaction, _ostree_repo_auto_transaction_unref);
G_END_DECLS G_END_DECLS

View File

@ -711,10 +711,9 @@ ostree_repo_auto_lock_cleanup (OstreeRepoAutoLock *auto_lock)
} }
} }
/** /**
* _ostree_repo_auto_transaction_start: * _ostree_repo_auto_transaction_start:
* @repo: an #OsreeRepo object * @repo: (not nullable): an #OsreeRepo object
* @cancellable: Cancellable * @cancellable: Cancellable
* @error: a #GError * @error: a #GError
* *
@ -734,6 +733,7 @@ _ostree_repo_auto_transaction_start (OstreeRepo *repo,
return NULL; return NULL;
OstreeRepoAutoTransaction *txn = g_malloc(sizeof(OstreeRepoAutoTransaction)); OstreeRepoAutoTransaction *txn = g_malloc(sizeof(OstreeRepoAutoTransaction));
txn->atomic_refcount = 1;
txn->repo = g_object_ref (repo); txn->repo = g_object_ref (repo);
return g_steal_pointer (&txn); return g_steal_pointer (&txn);
@ -741,7 +741,7 @@ _ostree_repo_auto_transaction_start (OstreeRepo *repo,
/** /**
* _ostree_repo_auto_transaction_abort: * _ostree_repo_auto_transaction_abort:
* @txn: an #OsreeRepoAutoTransaction guard * @txn: (not nullable): an #OsreeRepoAutoTransaction guard
* @cancellable: Cancellable * @cancellable: Cancellable
* @error: a #GError * @error: a #GError
* *
@ -770,7 +770,8 @@ _ostree_repo_auto_transaction_abort (OstreeRepoAutoTransaction *txn,
/** /**
* _ostree_repo_auto_transaction_commit: * _ostree_repo_auto_transaction_commit:
* @txn: an #OsreeRepoAutoTransaction guard * @txn: (not nullable): an #OsreeRepoAutoTransaction guard
* @out_stats: (out) (allow-none): transaction result statistics
* @cancellable: Cancellable * @cancellable: Cancellable
* @error: a #GError * @error: a #GError
* *
@ -799,30 +800,58 @@ _ostree_repo_auto_transaction_commit (OstreeRepoAutoTransaction *txn,
} }
/** /**
* _ostree_repo_auto_transaction_cleanup: * _ostree_repo_auto_transaction_ref:
* @p: pointer to an #OsreeRepoAutoTransaction guard * @txn: (not nullable): an #OsreeRepoAutoTransaction guard
* *
* Destroy a transaction guard. If the transaction has not yet been completed, * Return a new reference to the transaction guard.
* it gets aborted. *
* Returns: (transfer full) (not nullable): new transaction guard reference.
*/
OstreeRepoAutoTransaction *
_ostree_repo_auto_transaction_ref (OstreeRepoAutoTransaction *txn)
{
g_assert (txn != NULL);
gint refcount = g_atomic_int_add (&txn->atomic_refcount, 1);
g_assert (refcount > 1);
return txn;
}
/**
* _ostree_repo_auto_transaction_unref:
* @txn: (transfer full): an #OsreeRepoAutoTransaction guard
*
* Unreference a transaction guard. When the last reference is gone,
* if the transaction has not yet been completed, it gets aborted.
*/ */
void void
_ostree_repo_auto_transaction_cleanup (void *p) _ostree_repo_auto_transaction_unref (OstreeRepoAutoTransaction *txn)
{ {
if (p == NULL) if (txn == NULL)
return;
if (!g_atomic_int_dec_and_test (&txn->atomic_refcount))
return; return;
OstreeRepoAutoTransaction *txn = p;
// Auto-abort only if transaction has not already been aborted/committed. // Auto-abort only if transaction has not already been aborted/committed.
if (txn->repo != NULL) if (txn->repo != NULL)
{ {
g_autoptr(GError) error = NULL; g_autoptr(GError) error = NULL;
if (!_ostree_repo_auto_transaction_abort (txn, NULL, &error)) { if (!ostree_repo_abort_transaction (txn->repo, NULL, &error))
g_warning("Failed to auto-cleanup OSTree transaction: %s", error->message); g_warning("Failed to auto-cleanup OSTree transaction: %s", error->message);
g_clear_object (&txn->repo); g_clear_object (&txn->repo);
} }
}
g_free (txn);
return;
} }
G_DEFINE_BOXED_TYPE(OstreeRepoAutoTransaction, _ostree_repo_auto_transaction,
_ostree_repo_auto_transaction_ref,
_ostree_repo_auto_transaction_unref);
static GFile * static GFile *
get_remotes_d_dir (OstreeRepo *self, get_remotes_d_dir (OstreeRepo *self,
GFile *sysroot); GFile *sysroot);