Specifically the fields uid, gid, ctime, ino and dev are set to zero by JGit. Any stat checking by git will then need to check content, which may be very slow, particularly on Windows. Since mtime and size is typically enough we should allow the user to tell git to avoid checking these fields if they are set to zero in the index. This change introduces a core.ignorezerostat config option where the user can list the fields to ignore using the names above. Signed-off-by: Robin Rosenberg <robin.rosenberg@xxxxxxxxxx> --- Documentation/config.txt | 9 +++++++++ cache.h | 8 ++++++++ config.c | 25 +++++++++++++++++++++++++ environment.c | 1 + read-cache.c | 24 +++++++++++++++--------- 5 files changed, 58 insertions(+), 9 deletions(-) diff --git a/Documentation/config.txt b/Documentation/config.txt index d5809e0..7f34c94 100644 --- a/Documentation/config.txt +++ b/Documentation/config.txt @@ -235,6 +235,15 @@ core.trustctime:: crawlers and some backup systems). See linkgit:git-update-index[1]. True by default. +core.ignorezerostat:: + Affects the interpretation of some fields in the index. If + unset has no effect. When set to a comma separated list of fields, + each of the fields in the index will be excluded from comparison with + working tree if the index value is zero. The following fields + are recognzed: `uid', `gid', `ctime', `ino' and `dev'. When ctime is ignored + the setting of 'core.trustctime' is overridden by by this config + value. + core.quotepath:: The commands that output paths (e.g. 'ls-files', 'diff'), when not given the `-z` option, will quote diff --git a/cache.h b/cache.h index c257953..524e49a 100644 --- a/cache.h +++ b/cache.h @@ -536,6 +536,14 @@ extern int delete_ref(const char *, const unsigned char *sha1, int delopt); /* Environment bits from configuration mechanism */ extern int trust_executable_bit; extern int trust_ctime; +extern int check_nonzero_stat; +#define CHECK_NONZERO_STAT_UID (1<<0) +#define CHECK_NONZERO_STAT_GID (1<<1) +#define CHECK_NONZERO_STAT_CTIME (1<<2) +#define CHECK_NONZERO_STAT_INO (1<<3) +#define CHECK_NONZERO_STAT_DEV (1<<4) +#define CHECK_NONZERO_STAT_MASK ((1<<5)-1) + extern int quote_path_fully; extern int has_symlinks; extern int minimum_abbrev, default_abbrev; diff --git a/config.c b/config.c index 7b444b6..79485cd 100644 --- a/config.c +++ b/config.c @@ -566,6 +566,31 @@ static int git_default_core_config(const char *var, const char *value) trust_ctime = git_config_bool(var, value); return 0; } + if (!strcmp(var, "core.ignorezerostat")) { + char *copy, *tok; + git_config_string(©, "core.ignorezerostat", value); + check_nonzero_stat = CHECK_NONZERO_STAT_MASK; + tok = strtok(value, ","); + while (tok) { + if (strcasecmp(tok, "uid") == 0) + check_nonzero_stat &= ~CHECK_NONZERO_STAT_UID; + else if (strcasecmp(tok, "gid") == 0) + check_nonzero_stat &= ~CHECK_NONZERO_STAT_GID; + else if (strcasecmp(tok, "ctime") == 0) { + check_nonzero_stat &= ~CHECK_NONZERO_STAT_CTIME; + trust_ctime = 0; + } else if (strcasecmp(tok, "ino") == 0) + check_nonzero_stat &= ~CHECK_NONZERO_STAT_INO; + else if (strcasecmp(tok, "dev") == 0) + check_nonzero_stat &= ~CHECK_NONZERO_STAT_DEV; + else + die_bad_config(var); + tok = strtok(NULL, ","); + } + if (check_nonzero_stat >= CHECK_NONZERO_STAT_MASK) + die_bad_config(var); + free(copy); + } if (!strcmp(var, "core.quotepath")) { quote_path_fully = git_config_bool(var, value); diff --git a/environment.c b/environment.c index 85edd7f..e90b52f 100644 --- a/environment.c +++ b/environment.c @@ -13,6 +13,7 @@ int trust_executable_bit = 1; int trust_ctime = 1; +int check_nonzero_stat = CHECK_NONZERO_STAT_MASK; int has_symlinks = 1; int minimum_abbrev = 4, default_abbrev = 7; int ignore_case; diff --git a/read-cache.c b/read-cache.c index fda78bc..f7fe15d 100644 --- a/read-cache.c +++ b/read-cache.c @@ -197,8 +197,9 @@ static int ce_match_stat_basic(struct cache_entry *ce, struct stat *st) } if (ce->ce_mtime.sec != (unsigned int)st->st_mtime) changed |= MTIME_CHANGED; - if (trust_ctime && ce->ce_ctime.sec != (unsigned int)st->st_ctime) - changed |= CTIME_CHANGED; + if ((trust_ctime || ((check_nonzero_stat&CHECK_NONZERO_STAT_CTIME) && ce->ce_ctime.sec))) + if (ce->ce_ctime.sec != (unsigned int)st->st_ctime) + changed |= CTIME_CHANGED; #ifdef USE_NSEC if (ce->ce_mtime.nsec != ST_MTIME_NSEC(*st)) @@ -207,11 +208,15 @@ static int ce_match_stat_basic(struct cache_entry *ce, struct stat *st) changed |= CTIME_CHANGED; #endif - if (ce->ce_uid != (unsigned int) st->st_uid || - ce->ce_gid != (unsigned int) st->st_gid) - changed |= OWNER_CHANGED; - if (ce->ce_ino != (unsigned int) st->st_ino) - changed |= INODE_CHANGED; + if ((check_nonzero_stat&CHECK_NONZERO_STAT_UID) || ce->ce_uid) + if (ce->ce_uid != (unsigned int) st->st_uid) + changed |= OWNER_CHANGED; + if ((check_nonzero_stat&CHECK_NONZERO_STAT_GID) || ce->ce_gid) + if (ce->ce_gid != (unsigned int) st->st_gid) + changed |= OWNER_CHANGED; + if ((check_nonzero_stat&CHECK_NONZERO_STAT_INO) || ce->ce_ino) + if (ce->ce_ino != (unsigned int) st->st_ino) + changed |= INODE_CHANGED; #ifdef USE_STDEV /* @@ -219,8 +224,9 @@ static int ce_match_stat_basic(struct cache_entry *ce, struct stat *st) * clients will have different views of what "device" * the filesystem is on */ - if (ce->ce_dev != (unsigned int) st->st_dev) - changed |= INODE_CHANGED; + if ((check_nonzero_stat&CHECK_NONZERO_STAT_DEV) || ce->ce_dev) + if (ce->ce_dev != (unsigned int) st->st_dev) + changed |= INODE_CHANGED; #endif if (ce->ce_size != (unsigned int) st->st_size) -- 1.8.1.337.g63e8afb.dirty -- To unsubscribe from this list: send the line "unsubscribe git" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html