Skip to content

Commit 61c7711

Browse files
peffgitster
authored andcommitted
sha1-file: use loose object cache for quick existence check
In cases where we expect to ask has_sha1_file() about a lot of objects that we are not likely to have (e.g., during fetch negotiation), we already use OBJECT_INFO_QUICK to sacrifice accuracy (due to racing with a simultaneous write or repack) for speed (we avoid re-scanning the pack directory). However, even checking for loose objects can be expensive, as we will stat() each one. On many systems this cost isn't too noticeable, but stat() can be particularly slow on some operating systems, or due to network filesystems. Since the QUICK flag already tells us that we're OK with a slightly stale answer, we can use that as a cue to look in our in-memory cache of each object directory. That basically trades an in-memory binary search for a stat() call. Note that it is possible for this to actually be _slower_. We'll do a full readdir() to fill the cache, so if you have a very large number of loose objects and a very small number of lookups, that readdir() may end up more expensive. This shouldn't be a big deal in practice. If you have a large number of reachable loose objects, you'll already run into performance problems (which you should remedy by repacking). You may have unreachable objects which wouldn't otherwise impact performance. Usually these would go away with the prune step of "git gc", but they may be held for up to 2 weeks in the default configuration. So it comes down to how many such objects you might reasonably expect to have, how much slower is readdir() on N entries versus M stat() calls (and here we really care about the syscall backing readdir(), like getdents() on Linux, but I'll just call this readdir() below). If N is much smaller than M (a typical packed repo), we know this is a big win (few readdirs() followed by many uses of the resulting cache). When N and M are similar in size, it's also a win. We care about the latency of making a syscall, and readdir() should be giving us many values in a single call. How many? On Linux, running "strace -e getdents ls" shows a 32k buffer getting 512 entries per call (which is 64 bytes per entry; the name itself is 38 bytes, plus there are some other fields). So we can imagine that this is always a win as long as the number of loose objects in the repository is a factor of 500 less than the number of lookups you make. It's hard to auto-tune this because we don't generally know up front how many lookups we're going to do. But it's unlikely for this to perform significantly worse. Signed-off-by: Jeff King <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 3a2e082 commit 61c7711

File tree

2 files changed

+21
-0
lines changed

2 files changed

+21
-0
lines changed

object-store.h

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ struct object_directory {
1313
/*
1414
* Used to store the results of readdir(3) calls when we are OK
1515
* sacrificing accuracy due to races for speed. That includes
16+
* object existence with OBJECT_INFO_QUICK, as well as
1617
* our search for unique abbreviated hashes. Don't use it for tasks
1718
* requiring greater accuracy!
1819
*

sha1-file.c

Lines changed: 20 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -921,6 +921,24 @@ static int open_sha1_file(struct repository *r,
921921
return -1;
922922
}
923923

924+
static int quick_has_loose(struct repository *r,
925+
const unsigned char *sha1)
926+
{
927+
int subdir_nr = sha1[0];
928+
struct object_id oid;
929+
struct object_directory *odb;
930+
931+
hashcpy(oid.hash, sha1);
932+
933+
prepare_alt_odb(r);
934+
for (odb = r->objects->odb; odb; odb = odb->next) {
935+
odb_load_loose_cache(odb, subdir_nr);
936+
if (oid_array_lookup(&odb->loose_objects_cache, &oid) >= 0)
937+
return 1;
938+
}
939+
return 0;
940+
}
941+
924942
/*
925943
* Map the loose object at "path" if it is not NULL, or the path found by
926944
* searching for a loose object named "sha1".
@@ -1171,6 +1189,8 @@ static int sha1_loose_object_info(struct repository *r,
11711189
if (!oi->typep && !oi->type_name && !oi->sizep && !oi->contentp) {
11721190
const char *path;
11731191
struct stat st;
1192+
if (!oi->disk_sizep && (flags & OBJECT_INFO_QUICK))
1193+
return quick_has_loose(r, sha1) ? 0 : -1;
11741194
if (stat_sha1_file(r, sha1, &st, &path) < 0)
11751195
return -1;
11761196
if (oi->disk_sizep)

0 commit comments

Comments
 (0)