packfile: move the MRU list into the packfile store

Packfiles have two lists associated to them:

  - A list that keeps track of packfiles in the order that they were
    added to a packfile store.

  - A list that keeps track of packfiles in most-recently-used order so
    that packfiles that are more likely to contain a specific object are
    ordered towards the front.

Both of these lists are hosted by `struct packed_git` itself, So to
identify all packfiles in a repository you simply need to grab the first
packfile and then iterate the `->next` pointers or the MRU list. This
pattern has the problem that all packfiles are part of the same list,
regardless of whether or not they belong to the same object source.

With the upcoming pluggable object database effort this needs to change:
packfiles should be contained by a single object source, and reading an
object from any such packfile should use that source to look up the
object. Consequently, we need to break up the global lists of packfiles
into per-object-source lists.

A first step towards this goal is to move those lists out of `struct
packed_git` and into the packfile store. While the packfile store is
currently sitting on the `struct object_database` level, the intent is
to push it down one level into the `struct odb_source` in a subsequent
patch series.

Introduce a new `struct packfile_list` that is used to manage lists of
packfiles and use it to store the list of most-recently-used packfiles
in `struct packfile_store`. For now, the new list type is only used in a
single spot, but we'll expand its usage in subsequent patches.

Signed-off-by: Patrick Steinhardt <ps@pks.im>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Patrick Steinhardt
2025-10-30 11:38:39 +01:00
committed by Junio C Hamano
parent e78ab37054
commit f905a855b1
4 changed files with 104 additions and 18 deletions

View File

@@ -1706,8 +1706,8 @@ static int want_object_in_pack_mtime(const struct object_id *oid,
uint32_t found_mtime) uint32_t found_mtime)
{ {
int want; int want;
struct packfile_list_entry *e;
struct odb_source *source; struct odb_source *source;
struct list_head *pos;
if (!exclude && local) { if (!exclude && local) {
/* /*
@@ -1748,12 +1748,11 @@ static int want_object_in_pack_mtime(const struct object_id *oid,
} }
} }
list_for_each(pos, packfile_store_get_packs_mru(the_repository->objects->packfiles)) { for (e = the_repository->objects->packfiles->mru.head; e; e = e->next) {
struct packed_git *p = list_entry(pos, struct packed_git, mru); struct packed_git *p = e->pack;
want = want_object_in_pack_one(p, oid, exclude, found_pack, found_offset, found_mtime); want = want_object_in_pack_one(p, oid, exclude, found_pack, found_offset, found_mtime);
if (!exclude && want > 0) if (!exclude && want > 0)
list_move(&p->mru, packfile_list_prepend(&the_repository->objects->packfiles->mru, p);
packfile_store_get_packs_mru(the_repository->objects->packfiles));
if (want != -1) if (want != -1)
return want; return want;
} }

2
midx.c
View File

@@ -463,7 +463,7 @@ int prepare_midx_pack(struct multi_pack_index *m,
p = packfile_store_load_pack(r->objects->packfiles, p = packfile_store_load_pack(r->objects->packfiles,
pack_name.buf, m->source->local); pack_name.buf, m->source->local);
if (p) if (p)
list_add_tail(&p->mru, &r->objects->packfiles->mru); packfile_list_append(&m->source->odb->packfiles->mru, p);
strbuf_release(&pack_name); strbuf_release(&pack_name);
if (!p) { if (!p) {

View File

@@ -47,6 +47,80 @@ static size_t pack_mapped;
#define SZ_FMT PRIuMAX #define SZ_FMT PRIuMAX
static inline uintmax_t sz_fmt(size_t s) { return s; } static inline uintmax_t sz_fmt(size_t s) { return s; }
void packfile_list_clear(struct packfile_list *list)
{
struct packfile_list_entry *e, *next;
for (e = list->head; e; e = next) {
next = e->next;
free(e);
}
list->head = list->tail = NULL;
}
static struct packfile_list_entry *packfile_list_remove_internal(struct packfile_list *list,
struct packed_git *pack)
{
struct packfile_list_entry *e, *prev;
for (e = list->head, prev = NULL; e; prev = e, e = e->next) {
if (e->pack != pack)
continue;
if (prev)
prev->next = e->next;
if (list->head == e)
list->head = e->next;
if (list->tail == e)
list->tail = prev;
return e;
}
return NULL;
}
void packfile_list_remove(struct packfile_list *list, struct packed_git *pack)
{
free(packfile_list_remove_internal(list, pack));
}
void packfile_list_prepend(struct packfile_list *list, struct packed_git *pack)
{
struct packfile_list_entry *entry;
entry = packfile_list_remove_internal(list, pack);
if (!entry) {
entry = xmalloc(sizeof(*entry));
entry->pack = pack;
}
entry->next = list->head;
list->head = entry;
if (!list->tail)
list->tail = entry;
}
void packfile_list_append(struct packfile_list *list, struct packed_git *pack)
{
struct packfile_list_entry *entry;
entry = packfile_list_remove_internal(list, pack);
if (!entry) {
entry = xmalloc(sizeof(*entry));
entry->pack = pack;
}
entry->next = NULL;
if (list->tail) {
list->tail->next = entry;
list->tail = entry;
} else {
list->head = list->tail = entry;
}
}
void pack_report(struct repository *repo) void pack_report(struct repository *repo)
{ {
fprintf(stderr, fprintf(stderr,
@@ -995,10 +1069,10 @@ static void packfile_store_prepare_mru(struct packfile_store *store)
{ {
struct packed_git *p; struct packed_git *p;
INIT_LIST_HEAD(&store->mru); packfile_list_clear(&store->mru);
for (p = store->packs; p; p = p->next) for (p = store->packs; p; p = p->next)
list_add_tail(&p->mru, &store->mru); packfile_list_append(&store->mru, p);
} }
void packfile_store_prepare(struct packfile_store *store) void packfile_store_prepare(struct packfile_store *store)
@@ -1040,10 +1114,10 @@ struct packed_git *packfile_store_get_packs(struct packfile_store *store)
return store->packs; return store->packs;
} }
struct list_head *packfile_store_get_packs_mru(struct packfile_store *store) struct packfile_list_entry *packfile_store_get_packs_mru(struct packfile_store *store)
{ {
packfile_store_prepare(store); packfile_store_prepare(store);
return &store->mru; return store->mru.head;
} }
/* /*
@@ -2048,7 +2122,7 @@ static int fill_pack_entry(const struct object_id *oid,
int find_pack_entry(struct repository *r, const struct object_id *oid, struct pack_entry *e) int find_pack_entry(struct repository *r, const struct object_id *oid, struct pack_entry *e)
{ {
struct list_head *pos; struct packfile_list_entry *l;
packfile_store_prepare(r->objects->packfiles); packfile_store_prepare(r->objects->packfiles);
@@ -2059,10 +2133,11 @@ int find_pack_entry(struct repository *r, const struct object_id *oid, struct pa
if (!r->objects->packfiles->packs) if (!r->objects->packfiles->packs)
return 0; return 0;
list_for_each(pos, &r->objects->packfiles->mru) { for (l = r->objects->packfiles->mru.head; l; l = l->next) {
struct packed_git *p = list_entry(pos, struct packed_git, mru); struct packed_git *p = l->pack;
if (!p->multi_pack_index && fill_pack_entry(oid, e, p)) { if (!p->multi_pack_index && fill_pack_entry(oid, e, p)) {
list_move(&p->mru, &r->objects->packfiles->mru); packfile_list_prepend(&r->objects->packfiles->mru, p);
return 1; return 1;
} }
} }
@@ -2314,7 +2389,6 @@ struct packfile_store *packfile_store_new(struct object_database *odb)
struct packfile_store *store; struct packfile_store *store;
CALLOC_ARRAY(store, 1); CALLOC_ARRAY(store, 1);
store->odb = odb; store->odb = odb;
INIT_LIST_HEAD(&store->mru);
strmap_init(&store->packs_by_path); strmap_init(&store->packs_by_path);
return store; return store;
} }

View File

@@ -12,7 +12,6 @@ struct object_info;
struct packed_git { struct packed_git {
struct packed_git *next; struct packed_git *next;
struct list_head mru;
struct pack_window *windows; struct pack_window *windows;
off_t pack_size; off_t pack_size;
const void *index_data; const void *index_data;
@@ -52,6 +51,20 @@ struct packed_git {
char pack_name[FLEX_ARRAY]; /* more */ char pack_name[FLEX_ARRAY]; /* more */
}; };
struct packfile_list {
struct packfile_list_entry *head, *tail;
};
struct packfile_list_entry {
struct packfile_list_entry *next;
struct packed_git *pack;
};
void packfile_list_clear(struct packfile_list *list);
void packfile_list_remove(struct packfile_list *list, struct packed_git *pack);
void packfile_list_prepend(struct packfile_list *list, struct packed_git *pack);
void packfile_list_append(struct packfile_list *list, struct packed_git *pack);
/* /*
* A store that manages packfiles for a given object database. * A store that manages packfiles for a given object database.
*/ */
@@ -79,7 +92,7 @@ struct packfile_store {
} kept_cache; } kept_cache;
/* A most-recently-used ordered version of the packs list. */ /* A most-recently-used ordered version of the packs list. */
struct list_head mru; struct packfile_list mru;
/* /*
* A map of packfile names to packed_git structs for tracking which * A map of packfile names to packed_git structs for tracking which
@@ -153,7 +166,7 @@ struct packed_git *packfile_store_get_packs(struct packfile_store *store);
/* /*
* Get all packs in most-recently-used order. * Get all packs in most-recently-used order.
*/ */
struct list_head *packfile_store_get_packs_mru(struct packfile_store *store); struct packfile_list_entry *packfile_store_get_packs_mru(struct packfile_store *store);
/* /*
* Open the packfile and add it to the store if it isn't yet known. Returns * Open the packfile and add it to the store if it isn't yet known. Returns