summaryrefslogtreecommitdiffstats
path: root/config.c
diff options
context:
space:
mode:
authorJeff King <peff@peff.net>2015-04-16 10:51:18 +0200
committerJunio C Hamano <gitster@pobox.com>2015-04-16 17:15:05 +0200
commit260d408e32363a9e76d0fea3056563d4fb51f29e (patch)
treea062f4480e0761fb1b5bbdbf49f08fc42e38f6a8 /config.c
parentstrbuf_getwholeline: use getc_unlocked (diff)
downloadgit-260d408e32363a9e76d0fea3056563d4fb51f29e.tar.xz
git-260d408e32363a9e76d0fea3056563d4fb51f29e.zip
config: use getc_unlocked when reading from file
We read config files character-by-character from a stdio handle using fgetc(). This incurs significant locking overhead, even though we know that only one thread can possibly access the handle. We can speed this up by taking the lock ourselves, and then using getc_unlocked to read each character. On a silly pathological case: perl -le ' print "[core]"; print "key$_ = value$_" for (1..1000000) ' >input git config -f input core.key1 this dropped the time to run git-config from: real 0m0.263s user 0m0.260s sys 0m0.000s to: real 0m0.159s user 0m0.152s sys 0m0.004s for a savings of 39%. Most config files are not this big, but the savings should be proportional to the size of the file (i.e., we always save 39%, just of a much smaller number). Signed-off-by: Jeff King <peff@peff.net> Signed-off-by: Junio C Hamano <gitster@pobox.com>
Diffstat (limited to 'config.c')
-rw-r--r--config.c4
1 files changed, 3 insertions, 1 deletions
diff --git a/config.c b/config.c
index 66c0a51bce..8b297fc4e5 100644
--- a/config.c
+++ b/config.c
@@ -49,7 +49,7 @@ static struct config_set the_config_set;
static int config_file_fgetc(struct config_source *conf)
{
- return fgetc(conf->u.file);
+ return getc_unlocked(conf->u.file);
}
static int config_file_ungetc(int c, struct config_source *conf)
@@ -1088,7 +1088,9 @@ int git_config_from_file(config_fn_t fn, const char *filename, void *data)
f = fopen(filename, "r");
if (f) {
+ flockfile(f);
ret = do_config_from_file(fn, filename, filename, f, data);
+ funlockfile(f);
fclose(f);
}
return ret;