diff options
Diffstat (limited to 'builtin/update-index.c')
| -rw-r--r-- | builtin/update-index.c | 112 | 
1 files changed, 80 insertions, 32 deletions
| diff --git a/builtin/update-index.c b/builtin/update-index.c index e3a10d706d..e8c7fd4d49 100644 --- a/builtin/update-index.c +++ b/builtin/update-index.c @@ -12,6 +12,8 @@  #include "resolve-undo.h"  #include "parse-options.h"  #include "pathspec.h" +#include "dir.h" +#include "split-index.h"  /*   * Default to not allowing changes to the list of files. The @@ -54,8 +56,9 @@ static int mark_ce_flags(const char *path, int flag, int mark)  			active_cache[pos]->ce_flags |= flag;  		else  			active_cache[pos]->ce_flags &= ~flag; -		cache_tree_invalidate_path(active_cache_tree, path); -		active_cache_changed = 1; +		active_cache[pos]->ce_flags |= CE_UPDATE_IN_BASE; +		cache_tree_invalidate_path(&the_index, path); +		active_cache_changed |= CE_ENTRY_CHANGED;  		return 0;  	}  	return -1; @@ -266,44 +269,41 @@ static void chmod_path(int flip, const char *path)  	default:  		goto fail;  	} -	cache_tree_invalidate_path(active_cache_tree, path); -	active_cache_changed = 1; +	cache_tree_invalidate_path(&the_index, path); +	ce->ce_flags |= CE_UPDATE_IN_BASE; +	active_cache_changed |= CE_ENTRY_CHANGED;  	report("chmod %cx '%s'", flip, path);  	return;   fail:  	die("git update-index: cannot chmod %cx '%s'", flip, path);  } -static void update_one(const char *path, const char *prefix, int prefix_length) +static void update_one(const char *path)  { -	const char *p = prefix_path(prefix, prefix_length, path); -	if (!verify_path(p)) { +	if (!verify_path(path)) {  		fprintf(stderr, "Ignoring path %s\n", path); -		goto free_return; +		return;  	}  	if (mark_valid_only) { -		if (mark_ce_flags(p, CE_VALID, mark_valid_only == MARK_FLAG)) +		if (mark_ce_flags(path, CE_VALID, mark_valid_only == MARK_FLAG))  			die("Unable to mark file %s", path); -		goto free_return; +		return;  	}  	if (mark_skip_worktree_only) { -		if (mark_ce_flags(p, CE_SKIP_WORKTREE, mark_skip_worktree_only == MARK_FLAG)) +		if (mark_ce_flags(path, CE_SKIP_WORKTREE, mark_skip_worktree_only == MARK_FLAG))  			die("Unable to mark file %s", path); -		goto free_return; +		return;  	}  	if (force_remove) { -		if (remove_file_from_cache(p)) +		if (remove_file_from_cache(path))  			die("git update-index: unable to remove %s", path);  		report("remove '%s'", path); -		goto free_return; +		return;  	} -	if (process_path(p)) +	if (process_path(path))  		die("Unable to process path %s", path);  	report("add '%s'", path); - free_return: -	if (p < path || p > path + strlen(path)) -		free((char *)p);  }  static void read_index_info(int line_termination) @@ -563,8 +563,9 @@ static int do_reupdate(int ac, const char **av,  		const struct cache_entry *ce = active_cache[pos];  		struct cache_entry *old = NULL;  		int save_nr; +		char *path; -		if (ce_stage(ce) || !ce_path_match(ce, &pathspec)) +		if (ce_stage(ce) || !ce_path_match(ce, &pathspec, NULL))  			continue;  		if (has_head)  			old = read_one_ent(NULL, head_sha1, @@ -579,7 +580,9 @@ static int do_reupdate(int ac, const char **av,  		 * or worse yet 'allow_replace', active_nr may decrease.  		 */  		save_nr = active_nr; -		update_one(ce->name + prefix_length, prefix, prefix_length); +		path = xstrdup(ce->name); +		update_one(path); +		free(path);  		if (save_nr != active_nr)  			goto redo;  	} @@ -629,14 +632,45 @@ static int resolve_undo_clear_callback(const struct option *opt,  	return 0;  } +static int parse_new_style_cacheinfo(const char *arg, +				     unsigned int *mode, +				     unsigned char sha1[], +				     const char **path) +{ +	unsigned long ul; +	char *endp; + +	if (!arg) +		return -1; + +	errno = 0; +	ul = strtoul(arg, &endp, 8); +	if (errno || endp == arg || *endp != ',' || (unsigned int) ul != ul) +		return -1; /* not a new-style cacheinfo */ +	*mode = ul; +	endp++; +	if (get_sha1_hex(endp, sha1) || endp[40] != ',') +		return -1; +	*path = endp + 41; +	return 0; +} +  static int cacheinfo_callback(struct parse_opt_ctx_t *ctx,  				const struct option *opt, int unset)  {  	unsigned char sha1[20];  	unsigned int mode; +	const char *path; +	if (!parse_new_style_cacheinfo(ctx->argv[1], &mode, sha1, &path)) { +		if (add_cacheinfo(mode, sha1, path, 0)) +			die("git update-index: --cacheinfo cannot add %s", path); +		ctx->argv++; +		ctx->argc--; +		return 0; +	}  	if (ctx->argc <= 3) -		return error("option 'cacheinfo' expects three arguments"); +		return error("option 'cacheinfo' expects <mode>,<sha1>,<path>");  	if (strtoul_ui(*++ctx->argv, 8, &mode) ||  	    get_sha1_hex(*++ctx->argv, sha1) ||  	    add_cacheinfo(mode, sha1, *++ctx->argv, 0)) @@ -712,6 +746,7 @@ int cmd_update_index(int argc, const char **argv, const char *prefix)  	char set_executable_bit = 0;  	struct refresh_params refresh_args = {0, &has_errors};  	int lock_error = 0; +	int split_index = -1;  	struct lock_file *lock_file;  	struct parse_opt_ctx_t ctx;  	int parseopt_state = PARSE_OPT_UNKNOWN; @@ -740,9 +775,9 @@ int cmd_update_index(int argc, const char **argv, const char *prefix)  			PARSE_OPT_NOARG | PARSE_OPT_NONEG,  			really_refresh_callback},  		{OPTION_LOWLEVEL_CALLBACK, 0, "cacheinfo", NULL, -			N_("<mode> <object> <path>"), +			N_("<mode>,<object>,<path>"),  			N_("add the specified entry to the index"), -			PARSE_OPT_NOARG |	/* disallow --cacheinfo=<mode> form */ +			PARSE_OPT_NOARG | /* disallow --cacheinfo=<mode> form */  			PARSE_OPT_NONEG | PARSE_OPT_LITERAL_ARGHELP,  			(parse_opt_cb *) cacheinfo_callback},  		{OPTION_CALLBACK, 0, "chmod", &set_executable_bit, N_("(+/-)x"), @@ -794,6 +829,8 @@ int cmd_update_index(int argc, const char **argv, const char *prefix)  			resolve_undo_clear_callback},  		OPT_INTEGER(0, "index-version", &preferred_index_format,  			N_("write index in this format")), +		OPT_BOOL(0, "split-index", &split_index, +			N_("enable or disable split index")),  		OPT_END()  	}; @@ -836,11 +873,10 @@ int cmd_update_index(int argc, const char **argv, const char *prefix)  			setup_work_tree();  			p = prefix_path(prefix, prefix_length, path); -			update_one(p, NULL, 0); +			update_one(p);  			if (set_executable_bit)  				chmod_path(set_executable_bit, p); -			if (p < path || p > path + strlen(path)) -				free((char *)p); +			free((char *)p);  			ctx.argc--;  			ctx.argv++;  			break; @@ -862,7 +898,7 @@ int cmd_update_index(int argc, const char **argv, const char *prefix)  			    INDEX_FORMAT_LB, INDEX_FORMAT_UB);  		if (the_index.version != preferred_index_format) -			active_cache_changed = 1; +			active_cache_changed |= SOMETHING_CHANGED;  		the_index.version = preferred_index_format;  	} @@ -879,24 +915,36 @@ int cmd_update_index(int argc, const char **argv, const char *prefix)  				strbuf_swap(&buf, &nbuf);  			}  			p = prefix_path(prefix, prefix_length, buf.buf); -			update_one(p, NULL, 0); +			update_one(p);  			if (set_executable_bit)  				chmod_path(set_executable_bit, p); -			if (p < buf.buf || p > buf.buf + buf.len) -				free((char *)p); +			free((char *)p);  		}  		strbuf_release(&nbuf);  		strbuf_release(&buf);  	} +	if (split_index > 0) { +		init_split_index(&the_index); +		the_index.cache_changed |= SPLIT_INDEX_ORDERED; +	} else if (!split_index && the_index.split_index) { +		/* +		 * can't discard_split_index(&the_index); because that +		 * will destroy split_index->base->cache[], which may +		 * be shared with the_index.cache[]. So yeah we're +		 * leaking a bit here. +		 */ +		the_index.split_index = NULL; +		the_index.cache_changed |= SOMETHING_CHANGED; +	} +  	if (active_cache_changed) {  		if (newfd < 0) {  			if (refresh_args.flags & REFRESH_QUIET)  				exit(128);  			unable_to_lock_index_die(get_index_file(), lock_error);  		} -		if (write_cache(newfd, active_cache, active_nr) || -		    commit_locked_index(lock_file)) +		if (write_locked_index(&the_index, lock_file, COMMIT_LOCK))  			die("Unable to write new index file");  	} | 
