name-hash.c: remove cache entries instead of marking them CE_UNHASHED

The new hashmap implementation supports remove, so really remove unused
cache entries from the name hashmap instead of just marking them.

The CE_UNHASHED flag and CE_STATE_MASK are no longer needed.

Keep the CE_HASHED flag to prevent adding entries twice.

Signed-off-by: Karsten Blees <blees@dcon.de>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Karsten Blees 2013-11-14 20:22:27 +01:00 committed by Junio C Hamano
parent 8b013788a1
commit 419a597f64
4 changed files with 26 additions and 30 deletions

View File

@ -160,7 +160,6 @@ struct cache_entry {
#define CE_ADDED (1 << 19) #define CE_ADDED (1 << 19)
#define CE_HASHED (1 << 20) #define CE_HASHED (1 << 20)
#define CE_UNHASHED (1 << 21)
#define CE_WT_REMOVE (1 << 22) /* remove in work directory */ #define CE_WT_REMOVE (1 << 22) /* remove in work directory */
#define CE_CONFLICTED (1 << 23) #define CE_CONFLICTED (1 << 23)
@ -196,11 +195,10 @@ struct pathspec;
* Copy the sha1 and stat state of a cache entry from one to * Copy the sha1 and stat state of a cache entry from one to
* another. But we never change the name, or the hash state! * another. But we never change the name, or the hash state!
*/ */
#define CE_STATE_MASK (CE_HASHED | CE_UNHASHED)
static inline void copy_cache_entry(struct cache_entry *dst, static inline void copy_cache_entry(struct cache_entry *dst,
const struct cache_entry *src) const struct cache_entry *src)
{ {
unsigned int state = dst->ce_flags & CE_STATE_MASK; unsigned int state = dst->ce_flags & CE_HASHED;
/* Don't copy hash chain and name */ /* Don't copy hash chain and name */
memcpy(&dst->ce_stat_data, &src->ce_stat_data, memcpy(&dst->ce_stat_data, &src->ce_stat_data,
@ -208,7 +206,7 @@ static inline void copy_cache_entry(struct cache_entry *dst,
offsetof(struct cache_entry, ce_stat_data)); offsetof(struct cache_entry, ce_stat_data));
/* Restore the hash state */ /* Restore the hash state */
dst->ce_flags = (dst->ce_flags & ~CE_STATE_MASK) | state; dst->ce_flags = (dst->ce_flags & ~CE_HASHED) | state;
} }
static inline unsigned create_ce_flags(unsigned stage) static inline unsigned create_ce_flags(unsigned stage)

View File

@ -106,17 +106,29 @@ static void hash_index_entry(struct index_state *istate, struct cache_entry *ce)
hashmap_entry_init(ce, memihash(ce->name, ce_namelen(ce))); hashmap_entry_init(ce, memihash(ce->name, ce_namelen(ce)));
hashmap_add(&istate->name_hash, ce); hashmap_add(&istate->name_hash, ce);
if (ignore_case && !(ce->ce_flags & CE_UNHASHED)) if (ignore_case)
add_dir_entry(istate, ce); add_dir_entry(istate, ce);
} }
static int cache_entry_cmp(const struct cache_entry *ce1,
const struct cache_entry *ce2, const void *remove)
{
/*
* For remove_name_hash, find the exact entry (pointer equality); for
* index_name_exists, find all entries with matching hash code and
* decide whether the entry matches in same_name.
*/
return remove ? !(ce1 == ce2) : 0;
}
static void lazy_init_name_hash(struct index_state *istate) static void lazy_init_name_hash(struct index_state *istate)
{ {
int nr; int nr;
if (istate->name_hash_initialized) if (istate->name_hash_initialized)
return; return;
hashmap_init(&istate->name_hash, NULL, istate->cache_nr); hashmap_init(&istate->name_hash, (hashmap_cmp_fn) cache_entry_cmp,
istate->cache_nr);
hashmap_init(&istate->dir_hash, (hashmap_cmp_fn) dir_entry_cmp, 0); hashmap_init(&istate->dir_hash, (hashmap_cmp_fn) dir_entry_cmp, 0);
for (nr = 0; nr < istate->cache_nr; nr++) for (nr = 0; nr < istate->cache_nr; nr++)
hash_index_entry(istate, istate->cache[nr]); hash_index_entry(istate, istate->cache[nr]);
@ -125,31 +137,19 @@ static void lazy_init_name_hash(struct index_state *istate)
void add_name_hash(struct index_state *istate, struct cache_entry *ce) void add_name_hash(struct index_state *istate, struct cache_entry *ce)
{ {
/* if already hashed, add reference to directory entries */
if (ignore_case && (ce->ce_flags & CE_STATE_MASK) == CE_STATE_MASK)
add_dir_entry(istate, ce);
ce->ce_flags &= ~CE_UNHASHED;
if (istate->name_hash_initialized) if (istate->name_hash_initialized)
hash_index_entry(istate, ce); hash_index_entry(istate, ce);
} }
/*
* We don't actually *remove* it, we can just mark it invalid so that
* we won't find it in lookups.
*
* Not only would we have to search the lists (simple enough), but
* we'd also have to rehash other hash buckets in case this makes the
* hash bucket empty (common). So it's much better to just mark
* it.
*/
void remove_name_hash(struct index_state *istate, struct cache_entry *ce) void remove_name_hash(struct index_state *istate, struct cache_entry *ce)
{ {
/* if already hashed, release reference to directory entries */ if (!istate->name_hash_initialized || !(ce->ce_flags & CE_HASHED))
if (ignore_case && (ce->ce_flags & CE_STATE_MASK) == CE_HASHED) return;
remove_dir_entry(istate, ce); ce->ce_flags &= ~CE_HASHED;
hashmap_remove(&istate->name_hash, ce, ce);
ce->ce_flags |= CE_UNHASHED; if (ignore_case)
remove_dir_entry(istate, ce);
} }
static int slow_same_name(const char *name1, int len1, const char *name2, int len2) static int slow_same_name(const char *name1, int len1, const char *name2, int len2)
@ -220,10 +220,8 @@ struct cache_entry *index_file_exists(struct index_state *istate, const char *na
hashmap_entry_init(&key, memihash(name, namelen)); hashmap_entry_init(&key, memihash(name, namelen));
ce = hashmap_get(&istate->name_hash, &key, NULL); ce = hashmap_get(&istate->name_hash, &key, NULL);
while (ce) { while (ce) {
if (!(ce->ce_flags & CE_UNHASHED)) { if (same_name(ce, name, namelen, icase))
if (same_name(ce, name, namelen, icase)) return ce;
return ce;
}
ce = hashmap_get_next(&istate->name_hash, ce); ce = hashmap_get_next(&istate->name_hash, ce);
} }
return NULL; return NULL;

View File

@ -58,7 +58,7 @@ void rename_index_entry_at(struct index_state *istate, int nr, const char *new_n
new = xmalloc(cache_entry_size(namelen)); new = xmalloc(cache_entry_size(namelen));
copy_cache_entry(new, old); copy_cache_entry(new, old);
new->ce_flags &= ~CE_STATE_MASK; new->ce_flags &= ~CE_HASHED;
new->ce_namelen = namelen; new->ce_namelen = namelen;
memcpy(new->name, new_name, namelen + 1); memcpy(new->name, new_name, namelen + 1);

View File

@ -105,7 +105,7 @@ void setup_unpack_trees_porcelain(struct unpack_trees_options *opts,
static void do_add_entry(struct unpack_trees_options *o, struct cache_entry *ce, static void do_add_entry(struct unpack_trees_options *o, struct cache_entry *ce,
unsigned int set, unsigned int clear) unsigned int set, unsigned int clear)
{ {
clear |= CE_HASHED | CE_UNHASHED; clear |= CE_HASHED;
if (set & CE_REMOVE) if (set & CE_REMOVE)
set |= CE_WT_REMOVE; set |= CE_WT_REMOVE;