negotiator: add have_sent() interface

In a future change, we will introduce a capability to choose specific commit
OIDs as 'have's in fetch negotiation, with the ability to have the
negotiator choose more 'have's to increase coverage beyond that required
core set. The negotiator works to avoid emitting 'have's that can reach each
other, but that logic is hidden beneath the negotiator's iterator function
pointer ('next'). We need a way to communicate to the negotiator that we
have picked a 'have' so it could incorporate that into its logic.

Add a have_sent() method to the fetch_negotiator interface. This is the
signal that allows the negotiator to track the commit as already shown and
can perform the proper bookkeeping to avoid emitting those objects or
anything they can reach.

For our non-trivial negotiators, it is sufficient to mark these commits as
common, so the implementation is quite simple. This logic will be exercised
in the next change.

Reviewed-by: Matthew John Cheetham <mjcheetham@outlook.com>
Signed-off-by: Derrick Stolee <stolee@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
main
Derrick Stolee 2026-05-19 16:24:52 +00:00 committed by Junio C Hamano
parent 8bb252f86c
commit 22b2f3d2a3
4 changed files with 32 additions and 0 deletions

View File

@ -47,6 +47,15 @@ struct fetch_negotiator {
*/
int (*ack)(struct fetch_negotiator *, struct commit *);

/*
* Inform the negotiator that this commit has already been sent as
* a "have" line outside of the negotiator's control. The negotiator
* should avoid outputting it from next() and may use it to optimize
* further negotiation (e.g., by treating it and its ancestors as
* common).
*/
void (*have_sent)(struct fetch_negotiator *, struct commit *);

void (*release)(struct fetch_negotiator *);

/* internal use */

View File

@ -175,6 +175,13 @@ static int ack(struct fetch_negotiator *n, struct commit *c)
return known_to_be_common;
}

static void have_sent(struct fetch_negotiator *n, struct commit *c)
{
if (repo_parse_commit(the_repository, c))
return;
mark_common(n->data, c, 0, 0);
}

static void release(struct fetch_negotiator *n)
{
clear_prio_queue(&((struct negotiation_state *)n->data)->rev_list);
@ -188,6 +195,7 @@ void default_negotiator_init(struct fetch_negotiator *negotiator)
negotiator->add_tip = add_tip;
negotiator->next = next;
negotiator->ack = ack;
negotiator->have_sent = have_sent;
negotiator->release = release;
negotiator->data = CALLOC_ARRAY(ns, 1);
ns->rev_list.compare = compare_commits_by_commit_date;

View File

@ -29,6 +29,12 @@ static int ack(struct fetch_negotiator *n UNUSED, struct commit *c UNUSED)
return 0;
}

static void have_sent(struct fetch_negotiator *n UNUSED,
struct commit *c UNUSED)
{
/* nothing to do */
}

static void release(struct fetch_negotiator *n UNUSED)
{
/* nothing to release */
@ -40,6 +46,7 @@ void noop_negotiator_init(struct fetch_negotiator *negotiator)
negotiator->add_tip = add_tip;
negotiator->next = next;
negotiator->ack = ack;
negotiator->have_sent = have_sent;
negotiator->release = release;
negotiator->data = NULL;
}

View File

@ -243,6 +243,13 @@ static int ack(struct fetch_negotiator *n, struct commit *c)
return known_to_be_common;
}

static void have_sent(struct fetch_negotiator *n, struct commit *c)
{
if (repo_parse_commit(the_repository, c))
return;
mark_common(n->data, c);
}

static void release(struct fetch_negotiator *n)
{
struct data *data = n->data;
@ -259,6 +266,7 @@ void skipping_negotiator_init(struct fetch_negotiator *negotiator)
negotiator->add_tip = add_tip;
negotiator->next = next;
negotiator->ack = ack;
negotiator->have_sent = have_sent;
negotiator->release = release;
negotiator->data = CALLOC_ARRAY(data, 1);
data->rev_list.compare = compare;