]> git.ipfire.org Git - thirdparty/git.git/commitdiff
packfile: move the MRU list into the packfile store
authorPatrick Steinhardt <ps@pks.im>
Thu, 30 Oct 2025 10:38:39 +0000 (11:38 +0100)
committerJunio C Hamano <gitster@pobox.com>
Thu, 30 Oct 2025 14:09:52 +0000 (07:09 -0700)
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>
builtin/pack-objects.c
midx.c
packfile.c
packfile.h

index b5454e5df137b44dfd1b75780b65b479f233c5a1..5348aebbe9f190a6bdf685cb500dda746ce24301 100644 (file)
@@ -1706,8 +1706,8 @@ static int want_object_in_pack_mtime(const struct object_id *oid,
                                     uint32_t found_mtime)
 {
        int want;
+       struct packfile_list_entry *e;
        struct odb_source *source;
-       struct list_head *pos;
 
        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)) {
-               struct packed_git *p = list_entry(pos, struct packed_git, mru);
+       for (e = the_repository->objects->packfiles->mru.head; e; e = e->next) {
+               struct packed_git *p = e->pack;
                want = want_object_in_pack_one(p, oid, exclude, found_pack, found_offset, found_mtime);
                if (!exclude && want > 0)
-                       list_move(&p->mru,
-                                 packfile_store_get_packs_mru(the_repository->objects->packfiles));
+                       packfile_list_prepend(&the_repository->objects->packfiles->mru, p);
                if (want != -1)
                        return want;
        }
diff --git a/midx.c b/midx.c
index 1d6269f957e7819fb62963cde191360422b29229..8022be9a45ecb981a54d9e0ed573d0b3d020d4a8 100644 (file)
--- a/midx.c
+++ b/midx.c
@@ -463,7 +463,7 @@ int prepare_midx_pack(struct multi_pack_index *m,
        p = packfile_store_load_pack(r->objects->packfiles,
                                     pack_name.buf, m->source->local);
        if (p)
-               list_add_tail(&p->mru, &r->objects->packfiles->mru);
+               packfile_list_append(&m->source->odb->packfiles->mru, p);
        strbuf_release(&pack_name);
 
        if (!p) {
index 04649e52920a1f35e2efb8e556991c2e019398c8..4d2d3b674f3fb00e6d550acc24e0a48fce85d507 100644 (file)
@@ -47,6 +47,80 @@ static size_t pack_mapped;
 #define SZ_FMT PRIuMAX
 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)
 {
        fprintf(stderr,
@@ -995,10 +1069,10 @@ static void packfile_store_prepare_mru(struct packfile_store *store)
 {
        struct packed_git *p;
 
-       INIT_LIST_HEAD(&store->mru);
+       packfile_list_clear(&store->mru);
 
        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)
@@ -1040,10 +1114,10 @@ struct packed_git *packfile_store_get_packs(struct packfile_store *store)
        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);
-       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)
 {
-       struct list_head *pos;
+       struct packfile_list_entry *l;
 
        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)
                return 0;
 
-       list_for_each(pos, &r->objects->packfiles->mru) {
-               struct packed_git *p = list_entry(pos, struct packed_git, mru);
+       for (l = r->objects->packfiles->mru.head; l; l = l->next) {
+               struct packed_git *p = l->pack;
+
                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;
                }
        }
@@ -2314,7 +2389,6 @@ struct packfile_store *packfile_store_new(struct object_database *odb)
        struct packfile_store *store;
        CALLOC_ARRAY(store, 1);
        store->odb = odb;
-       INIT_LIST_HEAD(&store->mru);
        strmap_init(&store->packs_by_path);
        return store;
 }
index 9da7f14317b02ca78111cd422996167b0b41a8e0..39ed1073e4ad79bb7cdf5426328bab53e7650c41 100644 (file)
@@ -12,7 +12,6 @@ struct object_info;
 
 struct packed_git {
        struct packed_git *next;
-       struct list_head mru;
        struct pack_window *windows;
        off_t pack_size;
        const void *index_data;
@@ -52,6 +51,20 @@ struct packed_git {
        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.
  */
@@ -79,7 +92,7 @@ struct packfile_store {
        } kept_cache;
 
        /* 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
@@ -153,7 +166,7 @@ struct packed_git *packfile_store_get_packs(struct packfile_store *store);
 /*
  * 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