* [PATCH v4 0/3] fetch --prune performance problem
@ 2025-07-02 1:12 Phil Hord
2025-07-02 1:12 ` [PATCH v4 1/3] fetch-prune: optimize dangling-ref reporting Phil Hord
` (3 more replies)
0 siblings, 4 replies; 8+ messages in thread
From: Phil Hord @ 2025-07-02 1:12 UTC (permalink / raw)
To: gitster; +Cc: peff, git, Jacob Keller, Phil Hord
From: Phil Hord <phil.hord@gmail.com>
`git fetch --prune` runs in O(N^2) time normally. This happens because the code
iterates over each ref to be pruned to display its status. In a repo with
174,000 refs, where I was pruning 15,000 refs, the current code made 2.6 billion
calls to strcmp and consumed 470 seconds of CPU. After this change, the same
operation completes in under 1 second.
The loop looks like this:
for p in prune_refs { for ref in all_refs { if p == ref { ... }}}
That loop runs only to check for and report newly dangling refs. A workaround to
avoid this slowness is to run with `-q` to bypass this check.
There is similar check/report functionality in `git remote prune`, but it uses a
more efficient method to check for dangling refs. prune_refs is first sorted, so
it can be searched in O(logN), so this loop is O(N*logN).
for ref in all_refs { if ref in prune_refs { ... }}
We can use that function instead, with some minor cleanup to the output to deal
with the ordering being changed.
This patch version only adds the deleted branch name to the output of the dangling
sym refs since the ordering has changed. This is only a minor cleanup and was
not actually needed since, for example, `git origin prune` already did not
mind losing track of this information in its output. But now it is improved
to be more explicit.
V3 forgot to include the first commit in the series (I forgot it grew).
So here's V4.
Phil Hord (3):
fetch-prune: optimize dangling-ref reporting
refs: remove old refs_warn_dangling_symref
clean up interface for refs_warn_dangling_symrefs
builtin/fetch.c | 19 ++++++++-----------
builtin/remote.c | 5 +----
refs.c | 34 ++++++++++++----------------------
refs.h | 5 ++---
4 files changed, 23 insertions(+), 40 deletions(-)
--
2.50.0.149.g1bab277d62
^ permalink raw reply [flat|nested] 8+ messages in thread
* [PATCH v4 1/3] fetch-prune: optimize dangling-ref reporting
2025-07-02 1:12 [PATCH v4 0/3] fetch --prune performance problem Phil Hord
@ 2025-07-02 1:12 ` Phil Hord
2025-07-02 1:12 ` [PATCH v4 2/3] refs: remove old refs_warn_dangling_symref Phil Hord
` (2 subsequent siblings)
3 siblings, 0 replies; 8+ messages in thread
From: Phil Hord @ 2025-07-02 1:12 UTC (permalink / raw)
To: gitster; +Cc: peff, git, Jacob Keller, Phil Hord
From: Phil Hord <phil.hord@gmail.com>
When pruning during `git fetch` we check each pruned ref against the
ref_store one at a time to decide whether to report it as dangling.
This causes every local ref to be scanned for each ref being pruned.
If there are N refs in the repo and M refs being pruned, this code is
O(M*N). However, `git remote prune` uses a very similar function that
is only O(N*log(M)).
Remove the wasteful ref scanning for each pruned ref and use the faster
version already available in refs_warn_dangling_symrefs. Change the
message to include the original refname since the message is no longer
printed immediately after the line that did just print the refname.
In a repo with 126,000 refs, where I was pruning 28,000 refs, this
code made about 3.6 billion calls to strcmp and consumed 410 seconds
of CPU. (Invariably in that time, my remote would timeout and the
fetch would fail anyway.)
After this change, the same operation completes in under a second.
Signed-off-by: Phil Hord <phil.hord@gmail.com>
Reviewed-by: Jacob Keller <jacob.e.keller@intel.com>
---
builtin/fetch.c | 20 ++++++++++----------
builtin/remote.c | 4 ++--
refs.c | 2 +-
3 files changed, 13 insertions(+), 13 deletions(-)
diff --git a/builtin/fetch.c b/builtin/fetch.c
index d48262bdc717..04d10c9e781a 100644
--- a/builtin/fetch.c
+++ b/builtin/fetch.c
@@ -1383,9 +1383,13 @@ static int prune_refs(struct display_state *display_state,
int result = 0;
struct ref *ref, *stale_refs = get_stale_heads(rs, ref_map);
struct strbuf err = STRBUF_INIT;
+ struct string_list refnames = STRING_LIST_INIT_NODUP;
const char *dangling_msg = dry_run
- ? _(" (%s will become dangling)")
- : _(" (%s has become dangling)");
+ ? _(" %s will become dangling after %s is deleted")
+ : _(" %s has become dangling after %s was deleted");
+
+ for (ref = stale_refs; ref; ref = ref->next)
+ string_list_append(&refnames, ref->name);
if (!dry_run) {
if (transaction) {
@@ -1396,15 +1400,9 @@ static int prune_refs(struct display_state *display_state,
goto cleanup;
}
} else {
- struct string_list refnames = STRING_LIST_INIT_NODUP;
-
- for (ref = stale_refs; ref; ref = ref->next)
- string_list_append(&refnames, ref->name);
-
result = refs_delete_refs(get_main_ref_store(the_repository),
"fetch: prune", &refnames,
0);
- string_list_clear(&refnames, 0);
}
}
@@ -1416,12 +1414,14 @@ static int prune_refs(struct display_state *display_state,
_("(none)"), ref->name,
&ref->new_oid, &ref->old_oid,
summary_width);
- refs_warn_dangling_symref(get_main_ref_store(the_repository),
- stderr, dangling_msg, ref->name);
}
+ string_list_sort(&refnames);
+ refs_warn_dangling_symrefs(get_main_ref_store(the_repository),
+ stderr, dangling_msg, &refnames);
}
cleanup:
+ string_list_clear(&refnames, 0);
strbuf_release(&err);
free_refs(stale_refs);
return result;
diff --git a/builtin/remote.c b/builtin/remote.c
index 0d6755bcb71e..4de7dd373ae5 100644
--- a/builtin/remote.c
+++ b/builtin/remote.c
@@ -1522,8 +1522,8 @@ static int prune_remote(const char *remote, int dry_run)
struct string_list refs_to_prune = STRING_LIST_INIT_NODUP;
struct string_list_item *item;
const char *dangling_msg = dry_run
- ? _(" %s will become dangling!")
- : _(" %s has become dangling!");
+ ? _(" %s will become dangling after %s is deleted!")
+ : _(" %s has become dangling after %s was deleted!");
get_remote_ref_states(remote, &states, GET_REF_STATES);
diff --git a/refs.c b/refs.c
index dce5c49ca2ba..651fb2d41299 100644
--- a/refs.c
+++ b/refs.c
@@ -461,7 +461,7 @@ static int warn_if_dangling_symref(const char *refname, const char *referent UNU
return 0;
}
- fprintf(d->fp, d->msg_fmt, refname);
+ fprintf(d->fp, d->msg_fmt, refname, resolves_to);
fputc('\n', d->fp);
return 0;
}
--
2.50.0.149.g1bab277d62
^ permalink raw reply related [flat|nested] 8+ messages in thread
* [PATCH v4 2/3] refs: remove old refs_warn_dangling_symref
2025-07-02 1:12 [PATCH v4 0/3] fetch --prune performance problem Phil Hord
2025-07-02 1:12 ` [PATCH v4 1/3] fetch-prune: optimize dangling-ref reporting Phil Hord
@ 2025-07-02 1:12 ` Phil Hord
2025-07-02 1:12 ` [PATCH v4 3/3] clean up interface for refs_warn_dangling_symrefs Phil Hord
2025-07-07 22:43 ` [PATCH v4 0/3] fetch --prune performance problem Junio C Hamano
3 siblings, 0 replies; 8+ messages in thread
From: Phil Hord @ 2025-07-02 1:12 UTC (permalink / raw)
To: gitster; +Cc: peff, git, Jacob Keller, Phil Hord
From: Phil Hord <phil.hord@gmail.com>
The dangling warning function that takes a single ref to search for
is no longer used. Remove it.
Signed-off-by: Phil Hord <phil.hord@gmail.com>
---
refs.c | 17 +----------------
refs.h | 2 --
2 files changed, 1 insertion(+), 18 deletions(-)
diff --git a/refs.c b/refs.c
index 651fb2d41299..07197c239e33 100644
--- a/refs.c
+++ b/refs.c
@@ -438,7 +438,6 @@ static int for_each_filter_refs(const char *refname, const char *referent,
struct warn_if_dangling_data {
struct ref_store *refs;
FILE *fp;
- const char *refname;
const struct string_list *refnames;
const char *msg_fmt;
};
@@ -455,9 +454,7 @@ static int warn_if_dangling_symref(const char *refname, const char *referent UNU
resolves_to = refs_resolve_ref_unsafe(d->refs, refname, 0, NULL, NULL);
if (!resolves_to
- || (d->refname
- ? strcmp(resolves_to, d->refname)
- : !string_list_has_string(d->refnames, resolves_to))) {
+ || !string_list_has_string(d->refnames, resolves_to)) {
return 0;
}
@@ -466,18 +463,6 @@ static int warn_if_dangling_symref(const char *refname, const char *referent UNU
return 0;
}
-void refs_warn_dangling_symref(struct ref_store *refs, FILE *fp,
- const char *msg_fmt, const char *refname)
-{
- struct warn_if_dangling_data data = {
- .refs = refs,
- .fp = fp,
- .refname = refname,
- .msg_fmt = msg_fmt,
- };
- refs_for_each_rawref(refs, warn_if_dangling_symref, &data);
-}
-
void refs_warn_dangling_symrefs(struct ref_store *refs, FILE *fp,
const char *msg_fmt, const struct string_list *refnames)
{
diff --git a/refs.h b/refs.h
index 46a6008e07f2..07f21824d480 100644
--- a/refs.h
+++ b/refs.h
@@ -452,8 +452,6 @@ static inline const char *has_glob_specials(const char *pattern)
return strpbrk(pattern, "?*[");
}
-void refs_warn_dangling_symref(struct ref_store *refs, FILE *fp,
- const char *msg_fmt, const char *refname);
void refs_warn_dangling_symrefs(struct ref_store *refs, FILE *fp,
const char *msg_fmt, const struct string_list *refnames);
--
2.50.0.149.g1bab277d62
^ permalink raw reply related [flat|nested] 8+ messages in thread
* [PATCH v4 3/3] clean up interface for refs_warn_dangling_symrefs
2025-07-02 1:12 [PATCH v4 0/3] fetch --prune performance problem Phil Hord
2025-07-02 1:12 ` [PATCH v4 1/3] fetch-prune: optimize dangling-ref reporting Phil Hord
2025-07-02 1:12 ` [PATCH v4 2/3] refs: remove old refs_warn_dangling_symref Phil Hord
@ 2025-07-02 1:12 ` Phil Hord
2025-07-08 1:35 ` Jeff King
2025-07-07 22:43 ` [PATCH v4 0/3] fetch --prune performance problem Junio C Hamano
3 siblings, 1 reply; 8+ messages in thread
From: Phil Hord @ 2025-07-02 1:12 UTC (permalink / raw)
To: gitster; +Cc: peff, git, Jacob Keller, Phil Hord
From: Phil Hord <phil.hord@gmail.com>
The refs_warn_dangling_symrefs interface is a bit fragile as it passes
in printf-formatting strings with expectations about the number of
arguments. This patch series made it worse by adding a 2nd positional
argument. But there are only two call sites, and they both use almost
identical display options.
Make this safer by moving the format strings into the function that uses
them to make it easier to see when the arguments don't match. Pass a
prefix string and a dry_run flag so the decision logic can be handled
where needed.
Signed-off-by: Phil Hord <phil.hord@gmail.com>
---
builtin/fetch.c | 5 +----
builtin/remote.c | 5 +----
refs.c | 17 +++++++++++------
refs.h | 3 ++-
4 files changed, 15 insertions(+), 15 deletions(-)
diff --git a/builtin/fetch.c b/builtin/fetch.c
index 04d10c9e781a..fc72f2119c56 100644
--- a/builtin/fetch.c
+++ b/builtin/fetch.c
@@ -1384,9 +1384,6 @@ static int prune_refs(struct display_state *display_state,
struct ref *ref, *stale_refs = get_stale_heads(rs, ref_map);
struct strbuf err = STRBUF_INIT;
struct string_list refnames = STRING_LIST_INIT_NODUP;
- const char *dangling_msg = dry_run
- ? _(" %s will become dangling after %s is deleted")
- : _(" %s has become dangling after %s was deleted");
for (ref = stale_refs; ref; ref = ref->next)
string_list_append(&refnames, ref->name);
@@ -1417,7 +1414,7 @@ static int prune_refs(struct display_state *display_state,
}
string_list_sort(&refnames);
refs_warn_dangling_symrefs(get_main_ref_store(the_repository),
- stderr, dangling_msg, &refnames);
+ stderr, " ", dry_run, &refnames);
}
cleanup:
diff --git a/builtin/remote.c b/builtin/remote.c
index 4de7dd373ae5..f672799e0d92 100644
--- a/builtin/remote.c
+++ b/builtin/remote.c
@@ -1521,9 +1521,6 @@ static int prune_remote(const char *remote, int dry_run)
struct ref_states states = REF_STATES_INIT;
struct string_list refs_to_prune = STRING_LIST_INIT_NODUP;
struct string_list_item *item;
- const char *dangling_msg = dry_run
- ? _(" %s will become dangling after %s is deleted!")
- : _(" %s has become dangling after %s was deleted!");
get_remote_ref_states(remote, &states, GET_REF_STATES);
@@ -1555,7 +1552,7 @@ static int prune_remote(const char *remote, int dry_run)
}
refs_warn_dangling_symrefs(get_main_ref_store(the_repository),
- stdout, dangling_msg, &refs_to_prune);
+ stdout, " ", dry_run, &refs_to_prune);
string_list_clear(&refs_to_prune, 0);
free_remote_ref_states(&states);
diff --git a/refs.c b/refs.c
index 07197c239e33..5602c18dbd5b 100644
--- a/refs.c
+++ b/refs.c
@@ -439,7 +439,8 @@ struct warn_if_dangling_data {
struct ref_store *refs;
FILE *fp;
const struct string_list *refnames;
- const char *msg_fmt;
+ const char *indent;
+ int dry_run;
};
static int warn_if_dangling_symref(const char *refname, const char *referent UNUSED,
@@ -447,7 +448,7 @@ static int warn_if_dangling_symref(const char *refname, const char *referent UNU
int flags, void *cb_data)
{
struct warn_if_dangling_data *d = cb_data;
- const char *resolves_to;
+ const char *resolves_to, *msg;
if (!(flags & REF_ISSYMREF))
return 0;
@@ -458,19 +459,23 @@ static int warn_if_dangling_symref(const char *refname, const char *referent UNU
return 0;
}
- fprintf(d->fp, d->msg_fmt, refname, resolves_to);
- fputc('\n', d->fp);
+ msg = d->dry_run
+ ? _("%s%s will become dangling after %s is deleted\n")
+ : _("%s%s has become dangling after %s was deleted\n");
+ fprintf(d->fp, msg, d->indent, refname, resolves_to);
return 0;
}
void refs_warn_dangling_symrefs(struct ref_store *refs, FILE *fp,
- const char *msg_fmt, const struct string_list *refnames)
+ const char *indent, int dry_run,
+ const struct string_list *refnames)
{
struct warn_if_dangling_data data = {
.refs = refs,
.fp = fp,
.refnames = refnames,
- .msg_fmt = msg_fmt,
+ .indent = indent,
+ .dry_run = dry_run,
};
refs_for_each_rawref(refs, warn_if_dangling_symref, &data);
}
diff --git a/refs.h b/refs.h
index 07f21824d480..25bed4d792e7 100644
--- a/refs.h
+++ b/refs.h
@@ -453,7 +453,8 @@ static inline const char *has_glob_specials(const char *pattern)
}
void refs_warn_dangling_symrefs(struct ref_store *refs, FILE *fp,
- const char *msg_fmt, const struct string_list *refnames);
+ const char *indent, int dry_run,
+ const struct string_list *refnames);
/*
* Flags for controlling behaviour of pack_refs()
--
2.50.0.149.g1bab277d62
^ permalink raw reply related [flat|nested] 8+ messages in thread
* Re: [PATCH v4 0/3] fetch --prune performance problem
2025-07-02 1:12 [PATCH v4 0/3] fetch --prune performance problem Phil Hord
` (2 preceding siblings ...)
2025-07-02 1:12 ` [PATCH v4 3/3] clean up interface for refs_warn_dangling_symrefs Phil Hord
@ 2025-07-07 22:43 ` Junio C Hamano
2025-07-08 1:00 ` Phil Hord
2025-07-08 1:36 ` Jeff King
3 siblings, 2 replies; 8+ messages in thread
From: Junio C Hamano @ 2025-07-07 22:43 UTC (permalink / raw)
To: Phil Hord; +Cc: peff, git, Jacob Keller
Phil Hord <phil.hord@gmail.com> writes:
> From: Phil Hord <phil.hord@gmail.com>
>
> `git fetch --prune` runs in O(N^2) time normally. This happens because the code
> iterates over each ref to be pruned to display its status. In a repo with
> 174,000 refs, where I was pruning 15,000 refs, the current code made 2.6 billion
> calls to strcmp and consumed 470 seconds of CPU. After this change, the same
> operation completes in under 1 second.
> ...
> V3 forgot to include the first commit in the series (I forgot it grew).
> So here's V4.
>
> Phil Hord (3):
> fetch-prune: optimize dangling-ref reporting
> refs: remove old refs_warn_dangling_symref
> clean up interface for refs_warn_dangling_symrefs
It seems that the thread has gone quiet. What's the status of this
topic?
Thanks.
^ permalink raw reply [flat|nested] 8+ messages in thread
* Re: [PATCH v4 0/3] fetch --prune performance problem
2025-07-07 22:43 ` [PATCH v4 0/3] fetch --prune performance problem Junio C Hamano
@ 2025-07-08 1:00 ` Phil Hord
2025-07-08 1:36 ` Jeff King
1 sibling, 0 replies; 8+ messages in thread
From: Phil Hord @ 2025-07-08 1:00 UTC (permalink / raw)
To: Junio C Hamano; +Cc: Jeff King, Git, Jacob Keller
On Mon, Jul 7, 2025, 3:43 PM Junio C Hamano <gitster@pobox.com> wrote:
>
> Phil Hord <phil.hord@gmail.com> writes:
>
> > From: Phil Hord <phil.hord@gmail.com>
> >
> > `git fetch --prune` runs in O(N^2) time normally. This happens because the code
> > iterates over each ref to be pruned to display its status. In a repo with
> > 174,000 refs, where I was pruning 15,000 refs, the current code made 2.6 billion
> > calls to strcmp and consumed 470 seconds of CPU. After this change, the same
> > operation completes in under 1 second.
> > ...
> > V3 forgot to include the first commit in the series (I forgot it grew).
> > So here's V4.
> >
> > Phil Hord (3):
> > fetch-prune: optimize dangling-ref reporting
> > refs: remove old refs_warn_dangling_symref
> > clean up interface for refs_warn_dangling_symrefs
>
> It seems that the thread has gone quiet. What's the status of this
> topic?
>
> Thanks.
LGTM
^ permalink raw reply [flat|nested] 8+ messages in thread
* Re: [PATCH v4 3/3] clean up interface for refs_warn_dangling_symrefs
2025-07-02 1:12 ` [PATCH v4 3/3] clean up interface for refs_warn_dangling_symrefs Phil Hord
@ 2025-07-08 1:35 ` Jeff King
0 siblings, 0 replies; 8+ messages in thread
From: Jeff King @ 2025-07-08 1:35 UTC (permalink / raw)
To: Phil Hord; +Cc: gitster, git, Jacob Keller
On Tue, Jul 01, 2025 at 06:12:15PM -0700, Phil Hord wrote:
> The refs_warn_dangling_symrefs interface is a bit fragile as it passes
> in printf-formatting strings with expectations about the number of
> arguments. This patch series made it worse by adding a 2nd positional
> argument. But there are only two call sites, and they both use almost
> identical display options.
>
> Make this safer by moving the format strings into the function that uses
> them to make it easier to see when the arguments don't match. Pass a
> prefix string and a dry_run flag so the decision logic can be handled
> where needed.
Thanks, I think the result is nicer. I have two comments, but I don't
think either will merit a re-roll.
> @@ -1384,9 +1384,6 @@ static int prune_refs(struct display_state *display_state,
> struct ref *ref, *stale_refs = get_stale_heads(rs, ref_map);
> struct strbuf err = STRBUF_INIT;
> struct string_list refnames = STRING_LIST_INIT_NODUP;
> - const char *dangling_msg = dry_run
> - ? _(" %s will become dangling after %s is deleted")
> - : _(" %s has become dangling after %s was deleted");
>
> for (ref = stale_refs; ref; ref = ref->next)
> string_list_append(&refnames, ref->name);
> @@ -1417,7 +1414,7 @@ static int prune_refs(struct display_state *display_state,
> }
> string_list_sort(&refnames);
> refs_warn_dangling_symrefs(get_main_ref_store(the_repository),
> - stderr, dangling_msg, &refnames);
> + stderr, " ", dry_run, &refnames);
> }
I had imagined passing in an "int indent", and not an arbitrary string.
But passing in the string is actually more flexible (it really could be
any prefix, not just an indentation). I think calling it "prefix" in the
actual function might be the more usual term here, but it's probably
just bike-shedding.
> - fprintf(d->fp, d->msg_fmt, refname, resolves_to);
> - fputc('\n', d->fp);
> + msg = d->dry_run
> + ? _("%s%s will become dangling after %s is deleted\n")
> + : _("%s%s has become dangling after %s was deleted\n");
> + fprintf(d->fp, msg, d->indent, refname, resolves_to);
Translators might find the extra "%s" at the beginning confusing without
context. I think you can do something like:
/* TRANSLATORS: The first %s is whitespace indentation. */
or similar. But maybe it would be more obvious as:
fputs(d->indent, d->fp);
fprintf(d->fp, msg, refname, resolves_to);
fputc('\n', d->fp);
? I dunno. Maybe putting it all together gives translators more options
(e.g., in a RTL language). I don't know much about translation.
Likewise on including the newline in the translated string. I think we
usually don't, just because we're mostly passing in strings for error(),
etc. But I don't know how much it matters.
-Peff
^ permalink raw reply [flat|nested] 8+ messages in thread
* Re: [PATCH v4 0/3] fetch --prune performance problem
2025-07-07 22:43 ` [PATCH v4 0/3] fetch --prune performance problem Junio C Hamano
2025-07-08 1:00 ` Phil Hord
@ 2025-07-08 1:36 ` Jeff King
1 sibling, 0 replies; 8+ messages in thread
From: Jeff King @ 2025-07-08 1:36 UTC (permalink / raw)
To: Junio C Hamano; +Cc: Phil Hord, git, Jacob Keller
On Mon, Jul 07, 2025 at 03:43:12PM -0700, Junio C Hamano wrote:
> Phil Hord <phil.hord@gmail.com> writes:
>
> > From: Phil Hord <phil.hord@gmail.com>
> >
> > `git fetch --prune` runs in O(N^2) time normally. This happens because the code
> > iterates over each ref to be pruned to display its status. In a repo with
> > 174,000 refs, where I was pruning 15,000 refs, the current code made 2.6 billion
> > calls to strcmp and consumed 470 seconds of CPU. After this change, the same
> > operation completes in under 1 second.
> > ...
> > V3 forgot to include the first commit in the series (I forgot it grew).
> > So here's V4.
> >
> > Phil Hord (3):
> > fetch-prune: optimize dangling-ref reporting
> > refs: remove old refs_warn_dangling_symref
> > clean up interface for refs_warn_dangling_symrefs
>
> It seems that the thread has gone quiet. What's the status of this
> topic?
v4 looks fine to me. I raised a few questions about the translation
strings, but I don't know if they're meaningful or not.
-Peff
^ permalink raw reply [flat|nested] 8+ messages in thread
end of thread, other threads:[~2025-07-08 1:36 UTC | newest]
Thread overview: 8+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2025-07-02 1:12 [PATCH v4 0/3] fetch --prune performance problem Phil Hord
2025-07-02 1:12 ` [PATCH v4 1/3] fetch-prune: optimize dangling-ref reporting Phil Hord
2025-07-02 1:12 ` [PATCH v4 2/3] refs: remove old refs_warn_dangling_symref Phil Hord
2025-07-02 1:12 ` [PATCH v4 3/3] clean up interface for refs_warn_dangling_symrefs Phil Hord
2025-07-08 1:35 ` Jeff King
2025-07-07 22:43 ` [PATCH v4 0/3] fetch --prune performance problem Junio C Hamano
2025-07-08 1:00 ` Phil Hord
2025-07-08 1:36 ` Jeff King
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox