On Tue, Aug 29, 2017 at 10:20:26AM +0200, Michael Haggerty wrote:
> `packed_ref_store` is going to want to store some transaction-wide
> data, so make a place for it.
That makes sense, although...
> diff --git a/refs/refs-internal.h b/refs/refs-internal.h
> index b02dc5a7e3..d7d344de73 100644
> --- a/refs/refs-internal.h
> +++ b/refs/refs-internal.h
> @@ -242,6 +242,7 @@ struct ref_transaction {
> size_t alloc;
> size_t nr;
> enum ref_transaction_state state;
> + void *backend_data;
> };
This is just one pointer. Once we start layering ref backends (and
already we're moving towards a "files" layer which sits atop loose and
packed backends, right?), how do we avoid backends stomping on each
other (or worse, dereferencing somebody else's data as their own
struct)?
I don't know that we necessarily need to answer that question right now,
but I'm worried that this pattern might need adjustment eventually.
I guess the hand-wavy answer is that whatever is doing the layering
would need to manage the pointers. So if you imagine that we had a
"union" backend that took two other arbitrary backends, it would
probably have something like:
struct union_backend_data {
void *data_a;
void *data_b;
}
and when it forwarded calls to the separate backends, it would give them
a view of the transaction with only their data. Something like:
void union_backend_foo(void *be, struct ref_transaction *transaction)
{
struct union_backend *me = be;
struct union_backend_data *my_data = transaction->backend_data;
/* "a" sees only it's data, and we remember any modifications */
transaction->backend_data = my_data->data_a;
me->backend_a->foo(me->backend_a, transaction);
my_data->data_a = transaction->backend_data;
/* ditto for "b" */
transaction->backend_data = my_data->data_b;
me->backend_b->foo(me->backend_b, transaction);
my_data->data_b = transaction->backend_data;
/* and then we restore our own view */
transaction->backend_data = my_data;
}
-Peff