| #include "cache.h" | 
 | #include "sha1-lookup.h" | 
 | #include "refs.h" | 
 | #include "commit.h" | 
 |  | 
 | static struct replace_object { | 
 | 	unsigned char sha1[2][20]; | 
 | } **replace_object; | 
 |  | 
 | static int replace_object_alloc, replace_object_nr; | 
 |  | 
 | static const unsigned char *replace_sha1_access(size_t index, void *table) | 
 | { | 
 | 	struct replace_object **replace = table; | 
 | 	return replace[index]->sha1[0]; | 
 | } | 
 |  | 
 | static int replace_object_pos(const unsigned char *sha1) | 
 | { | 
 | 	return sha1_pos(sha1, replace_object, replace_object_nr, | 
 | 			replace_sha1_access); | 
 | } | 
 |  | 
 | static int register_replace_object(struct replace_object *replace, | 
 | 				   int ignore_dups) | 
 | { | 
 | 	int pos = replace_object_pos(replace->sha1[0]); | 
 |  | 
 | 	if (0 <= pos) { | 
 | 		if (ignore_dups) | 
 | 			free(replace); | 
 | 		else { | 
 | 			free(replace_object[pos]); | 
 | 			replace_object[pos] = replace; | 
 | 		} | 
 | 		return 1; | 
 | 	} | 
 | 	pos = -pos - 1; | 
 | 	if (replace_object_alloc <= ++replace_object_nr) { | 
 | 		replace_object_alloc = alloc_nr(replace_object_alloc); | 
 | 		replace_object = xrealloc(replace_object, | 
 | 					  sizeof(*replace_object) * | 
 | 					  replace_object_alloc); | 
 | 	} | 
 | 	if (pos < replace_object_nr) | 
 | 		memmove(replace_object + pos + 1, | 
 | 			replace_object + pos, | 
 | 			(replace_object_nr - pos - 1) * | 
 | 			sizeof(*replace_object)); | 
 | 	replace_object[pos] = replace; | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int register_replace_ref(const char *refname, | 
 | 				const unsigned char *sha1, | 
 | 				int flag, void *cb_data) | 
 | { | 
 | 	/* Get sha1 from refname */ | 
 | 	const char *slash = strrchr(refname, '/'); | 
 | 	const char *hash = slash ? slash + 1 : refname; | 
 | 	struct replace_object *repl_obj = xmalloc(sizeof(*repl_obj)); | 
 |  | 
 | 	if (strlen(hash) != 40 || get_sha1_hex(hash, repl_obj->sha1[0])) { | 
 | 		free(repl_obj); | 
 | 		warning("bad replace ref name: %s", refname); | 
 | 		return 0; | 
 | 	} | 
 |  | 
 | 	/* Copy sha1 from the read ref */ | 
 | 	hashcpy(repl_obj->sha1[1], sha1); | 
 |  | 
 | 	/* Register new object */ | 
 | 	if (register_replace_object(repl_obj, 1)) | 
 | 		die("duplicate replace ref: %s", refname); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void prepare_replace_object(void) | 
 | { | 
 | 	static int replace_object_prepared; | 
 |  | 
 | 	if (replace_object_prepared) | 
 | 		return; | 
 |  | 
 | 	for_each_replace_ref(register_replace_ref, NULL); | 
 | 	replace_object_prepared = 1; | 
 | 	if (!replace_object_nr) | 
 | 		read_replace_refs = 0; | 
 | } | 
 |  | 
 | /* We allow "recursive" replacement. Only within reason, though */ | 
 | #define MAXREPLACEDEPTH 5 | 
 |  | 
 | const unsigned char *do_lookup_replace_object(const unsigned char *sha1) | 
 | { | 
 | 	int pos, depth = MAXREPLACEDEPTH; | 
 | 	const unsigned char *cur = sha1; | 
 |  | 
 | 	if (!read_replace_refs) | 
 | 		return sha1; | 
 |  | 
 | 	prepare_replace_object(); | 
 |  | 
 | 	/* Try to recursively replace the object */ | 
 | 	do { | 
 | 		if (--depth < 0) | 
 | 			die("replace depth too high for object %s", | 
 | 			    sha1_to_hex(sha1)); | 
 |  | 
 | 		pos = replace_object_pos(cur); | 
 | 		if (0 <= pos) | 
 | 			cur = replace_object[pos]->sha1[1]; | 
 | 	} while (0 <= pos); | 
 |  | 
 | 	return cur; | 
 | } |