]> git.ipfire.org Git - thirdparty/git.git/commitdiff
object-file: get rid of `the_repository` in `has_loose_object()`
authorPatrick Steinhardt <ps@pks.im>
Thu, 17 Jul 2025 04:56:29 +0000 (06:56 +0200)
committerJunio C Hamano <gitster@pobox.com>
Thu, 17 Jul 2025 05:16:13 +0000 (22:16 -0700)
We implicitly depend on `the_repository` in `has_loose_object()`.
Refactor the function to accept an `odb_source` as input that should be
checked for such a loose object.

This refactoring changes semantics of the function to not check the
whole object database for such a loose object anymore, but instead we
now only check that single source. Existing callers thus need to loop
through all sources manually now.

While this change may seem illogical at first, whether or not an object
exists in a specific format should be answered by the source using that
format. As such, we can eventually convert this into a generic function
`odb_source_has_object()` that simply checks whether a given object
exists in an object source. And as we will know about the format that
any given source uses it allows us to derive whether the object exists
in a given format.

This change also makes `has_loose_object_nonlocal()` obsolete. The only
caller of this function is adapted so that it skips the primary object
source.

Signed-off-by: Patrick Steinhardt <ps@pks.im>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
builtin/pack-objects.c
object-file.c
object-file.h

index 5781dec9808273a4db0387a3aa758801aee2e249..a44f0ce1c785edfa2c8e7fcefb13c1b6444a4e58 100644 (file)
@@ -1703,8 +1703,16 @@ static int want_object_in_pack_mtime(const struct object_id *oid,
        struct list_head *pos;
        struct multi_pack_index *m;
 
-       if (!exclude && local && has_loose_object_nonlocal(oid))
-               return 0;
+       if (!exclude && local) {
+               /*
+                * Note that we start iterating at `sources->next` so that we
+                * skip the local object source.
+                */
+               struct odb_source *source = the_repository->objects->sources->next;
+               for (; source; source = source->next)
+                       if (has_loose_object(source, oid))
+                               return 0;
+       }
 
        /*
         * If we already know the pack object lives in, start checks from that
@@ -3928,7 +3936,14 @@ static void add_cruft_object_entry(const struct object_id *oid, enum object_type
        } else {
                if (!want_object_in_pack_mtime(oid, 0, &pack, &offset, mtime))
                        return;
-               if (!pack && type == OBJ_BLOB && !has_loose_object(oid)) {
+               if (!pack && type == OBJ_BLOB) {
+                       struct odb_source *source = the_repository->objects->sources;
+                       int found = 0;
+
+                       for (; !found && source; source = source->next)
+                               if (has_loose_object(source, oid))
+                                       found = 1;
+
                        /*
                         * If a traversed tree has a missing blob then we want
                         * to avoid adding that missing object to our pack.
@@ -3942,7 +3957,8 @@ static void add_cruft_object_entry(const struct object_id *oid, enum object_type
                         * limited to "ensure non-tip blobs which don't exist in
                         * packs do exist via loose objects". Confused?
                         */
-                       return;
+                       if (!found)
+                               return;
                }
 
                entry = create_object_entry(oid, type, pack_name_hash_fn(name),
index bc395febc9d3455af5ba911d9dd84927b7d4845a..7aecaa3d2a034d761c08e0d530d1749b5c332022 100644 (file)
@@ -121,14 +121,10 @@ static int check_and_freshen(const struct object_id *oid, int freshen)
               check_and_freshen_nonlocal(oid, freshen);
 }
 
-int has_loose_object_nonlocal(const struct object_id *oid)
+int has_loose_object(struct odb_source *source,
+                    const struct object_id *oid)
 {
-       return check_and_freshen_nonlocal(oid, 0);
-}
-
-int has_loose_object(const struct object_id *oid)
-{
-       return check_and_freshen(oid, 0);
+       return check_and_freshen_odb(source, oid, 0);
 }
 
 int format_object_header(char *str, size_t size, enum object_type type,
@@ -1103,8 +1099,10 @@ int force_object_loose(const struct object_id *oid, time_t mtime)
        int hdrlen;
        int ret;
 
-       if (has_loose_object(oid))
-               return 0;
+       for (struct odb_source *source = repo->objects->sources; source; source = source->next)
+               if (has_loose_object(source, oid))
+                       return 0;
+
        oi.typep = &type;
        oi.sizep = &len;
        oi.contentp = &buf;
index 222ff2871a186c22a21955aaebe3a34927d9aab1..5b63a05ab515ed896c1e4e57273c47ac114afa48 100644 (file)
@@ -45,13 +45,12 @@ const char *odb_loose_path(struct odb_source *source,
                           const struct object_id *oid);
 
 /*
- * Return true iff an alternate object database has a loose object
+ * Return true iff an object database source has a loose object
  * with the specified name.  This function does not respect replace
  * references.
  */
-int has_loose_object_nonlocal(const struct object_id *);
-
-int has_loose_object(const struct object_id *);
+int has_loose_object(struct odb_source *source,
+                    const struct object_id *oid);
 
 void *map_loose_object(struct repository *r, const struct object_id *oid,
                       unsigned long *size);