diff options
| author | Junio C Hamano <gitster@pobox.com> | 2015-08-25 14:57:08 -0700 | 
|---|---|---|
| committer | Junio C Hamano <gitster@pobox.com> | 2015-08-25 14:57:08 -0700 | 
| commit | 32561f5dd39da3ac720f6778bc2e8aafed771eb5 (patch) | |
| tree | c066f6a534665f2db781e6223544f9bbd31fe99c /branch.c | |
| parent | 5d5be8167db3bcaf9836cac7fdd785c564484e06 (diff) | |
| parent | b02e8595e263e284e45a6262b92ef158b6966478 (diff) | |
| download | git-32561f5dd39da3ac720f6778bc2e8aafed771eb5.tar.gz | |
Merge branch 'dt/notes-multiple'
When linked worktree is used, simultaneous "notes merge" instances
for the same ref in refs/notes/* are prevented from stomping on
each other.
* dt/notes-multiple:
  notes: handle multiple worktrees
  worktrees: add find_shared_symref
Diffstat (limited to 'branch.c')
| -rw-r--r-- | branch.c | 46 | 
1 files changed, 34 insertions, 12 deletions
| @@ -311,21 +311,23 @@ void remove_branch_state(void)  	unlink(git_path_squash_msg());  } -static void check_linked_checkout(const char *branch, const char *id) +static char *find_linked_symref(const char *symref, const char *branch, +				const char *id)  {  	struct strbuf sb = STRBUF_INIT;  	struct strbuf path = STRBUF_INIT;  	struct strbuf gitdir = STRBUF_INIT; +	char *existing = NULL;  	/* -	 * $GIT_COMMON_DIR/HEAD is practically outside -	 * $GIT_DIR so resolve_ref_unsafe() won't work (it -	 * uses git_path). Parse the ref ourselves. +	 * $GIT_COMMON_DIR/$symref (e.g. HEAD) is practically outside +	 * $GIT_DIR so resolve_ref_unsafe() won't work (it uses +	 * git_path). Parse the ref ourselves.  	 */  	if (id) -		strbuf_addf(&path, "%s/worktrees/%s/HEAD", get_git_common_dir(), id); +		strbuf_addf(&path, "%s/worktrees/%s/%s", get_git_common_dir(), id, symref);  	else -		strbuf_addf(&path, "%s/HEAD", get_git_common_dir()); +		strbuf_addf(&path, "%s/%s", get_git_common_dir(), symref);  	if (!strbuf_readlink(&sb, path.buf, 0)) {  		if (!starts_with(sb.buf, "refs/") || @@ -347,33 +349,53 @@ static void check_linked_checkout(const char *branch, const char *id)  		strbuf_rtrim(&gitdir);  	} else  		strbuf_addstr(&gitdir, get_git_common_dir()); -	skip_prefix(branch, "refs/heads/", &branch);  	strbuf_strip_suffix(&gitdir, ".git"); -	die(_("'%s' is already checked out at '%s'"), branch, gitdir.buf); + +	existing = strbuf_detach(&gitdir, NULL);  done:  	strbuf_release(&path);  	strbuf_release(&sb);  	strbuf_release(&gitdir); + +	return existing;  } -void die_if_checked_out(const char *branch) +char *find_shared_symref(const char *symref, const char *target)  {  	struct strbuf path = STRBUF_INIT;  	DIR *dir;  	struct dirent *d; +	char *existing; -	check_linked_checkout(branch, NULL); +	if ((existing = find_linked_symref(symref, target, NULL))) +		return existing;  	strbuf_addf(&path, "%s/worktrees", get_git_common_dir());  	dir = opendir(path.buf);  	strbuf_release(&path);  	if (!dir) -		return; +		return NULL;  	while ((d = readdir(dir)) != NULL) {  		if (!strcmp(d->d_name, ".") || !strcmp(d->d_name, ".."))  			continue; -		check_linked_checkout(branch, d->d_name); +		existing = find_linked_symref(symref, target, d->d_name); +		if (existing) +			goto done;  	} +done:  	closedir(dir); + +	return existing; +} + +void die_if_checked_out(const char *branch) +{ +	char *existing; + +	existing = find_shared_symref("HEAD", branch); +	if (existing) { +		skip_prefix(branch, "refs/heads/", &branch); +		die(_("'%s' is already checked out at '%s'"), branch, existing); +	}  } | 
