@ -1,4 +1,8 @@
#include "cache.h"
#include "cache.h"
#include "dir.h"
static int inside_git_dir = -1;
static int inside_work_tree = -1;
const char *prefix_path(const char *prefix, int len, const char *path)
const char *prefix_path(const char *prefix, int len, const char *path)
{
{
@ -170,100 +174,89 @@ static int is_git_directory(const char *suspect)
return 1;
return 1;
}
}
static int inside_git_dir = -1;
int is_inside_git_dir(void)
int is_inside_git_dir(void)
{
{
if (inside_git_dir >= 0)
if (inside_git_dir < 0)
return inside_git_dir;
inside_git_dir = is_inside_dir(get_git_dir());
die("BUG: is_inside_git_dir called before setup_git_directory");
return inside_git_dir;
}
}
static int inside_work_tree = -1;
int is_inside_work_tree(void)
int is_inside_work_tree(void)
{
{
if (inside_git_dir >= 0)
if (inside_work_tree < 0)
return inside_work_tree;
inside_work_tree = is_inside_dir(get_git_work_tree());
die("BUG: is_inside_work_tree called before setup_git_directory");
return inside_work_tree;
}
}
static char *gitworktree_config;
/*
* If no worktree was given, and we are outside of a default work tree,
static int git_setup_config(const char *var, const char *value)
* now is the time to set it.
*
* In other words, if the user calls git with something like
*
* git --git-dir=/some/where/else/.git bla
*
* default to /some/where/else as working directory; if the specified
* git-dir does not end in "/.git", the cwd is used as working directory.
*/
const char *set_work_tree(const char *dir)
{
{
if (!strcmp(var, "core.worktree")) {
char dir_buffer[PATH_MAX];
if (gitworktree_config)
static char buffer[PATH_MAX + 1], *rel = NULL;
strlcpy(gitworktree_config, value, PATH_MAX);
int len, postfix_len = strlen(DEFAULT_GIT_DIR_ENVIRONMENT) + 1;
return 0;
/* strip the variable 'dir' of the postfix "/.git" if it has it */
len = strlen(dir);
if (len > postfix_len && !strcmp(dir + len - postfix_len,
"/" DEFAULT_GIT_DIR_ENVIRONMENT)) {
strncpy(dir_buffer, dir, len - postfix_len);
/* are we inside the default work tree? */
rel = get_relative_cwd(buffer, sizeof(buffer), dir_buffer);
}
/* if rel is set, the cwd is _not_ the current working tree */
if (rel && *rel) {
if (!is_absolute_path(dir))
set_git_dir(make_absolute_path(dir));
dir = dir_buffer;
chdir(dir);
strcat(rel, "/");
inside_git_dir = 0;
} else {
rel = NULL;
dir = getcwd(buffer, sizeof(buffer));
}
}
return git_default_config(var, value);
git_work_tree_cfg = xstrdup(dir);
inside_work_tree = 1;
return rel;
}
}
/*
* We cannot decide in this function whether we are in the work tree or
* not, since the config can only be read _after_ this function was called.
*/
const char *setup_git_directory_gently(int *nongit_ok)
const char *setup_git_directory_gently(int *nongit_ok)
{
{
const char *work_tree_env = getenv(GIT_WORK_TREE_ENVIRONMENT);
static char cwd[PATH_MAX+1];
static char cwd[PATH_MAX+1];
char worktree[PATH_MAX+1], gitdir[PATH_MAX+1];
const char *gitdirenv;
const char *gitdirenv, *gitworktree;
int len, offset;
int wt_rel_gitdir = 0;
/*
* If GIT_DIR is set explicitly, we're not going
* to do any discovery, but we still do repository
* validation.
*/
gitdirenv = getenv(GIT_DIR_ENVIRONMENT);
gitdirenv = getenv(GIT_DIR_ENVIRONMENT);
if (!gitdirenv) {
if (gitdirenv) {
int len, offset;
if (PATH_MAX - 40 < strlen(gitdirenv))
die("'$%s' too big", GIT_DIR_ENVIRONMENT);
if (!getcwd(cwd, sizeof(cwd)-1))
if (is_git_directory(gitdirenv)) {
die("Unable to read current working directory");
if (!work_tree_env)
return set_work_tree(gitdirenv);
offset = len = strlen(cwd);
for (;;) {
if (is_git_directory(".git"))
break;
if (offset == 0) {
offset = -1;
break;
}
chdir("..");
while (cwd[--offset] != '/')
; /* do nothing */
}
if (offset >= 0) {
inside_work_tree = 1;
git_config(git_default_config);
if (offset == len) {
inside_git_dir = 0;
return NULL;
}
cwd[len++] = '/';
cwd[len] = '\0';
inside_git_dir = !prefixcmp(cwd + offset + 1, ".git/");
return cwd + offset + 1;
}
if (chdir(cwd))
die("Cannot come back to cwd");
if (!is_git_directory(".")) {
if (nongit_ok) {
*nongit_ok = 1;
return NULL;
}
die("Not a git repository");
}
setenv(GIT_DIR_ENVIRONMENT, cwd, 1);
gitdirenv = getenv(GIT_DIR_ENVIRONMENT);
if (!gitdirenv)
die("getenv after setenv failed");
}
if (PATH_MAX - 40 < strlen(gitdirenv)) {
if (nongit_ok) {
*nongit_ok = 1;
return NULL;
return NULL;
}
}
die("$%s too big", GIT_DIR_ENVIRONMENT);
}
if (!is_git_directory(gitdirenv)) {
if (nongit_ok) {
if (nongit_ok) {
*nongit_ok = 1;
*nongit_ok = 1;
return NULL;
return NULL;
@ -273,92 +266,53 @@ const char *setup_git_directory_gently(int *nongit_ok)
if (!getcwd(cwd, sizeof(cwd)-1))
if (!getcwd(cwd, sizeof(cwd)-1))
die("Unable to read current working directory");
die("Unable to read current working directory");
if (chdir(gitdirenv)) {
if (nongit_ok) {
*nongit_ok = 1;
return NULL;
}
die("Cannot change directory to $%s '%s'",
GIT_DIR_ENVIRONMENT, gitdirenv);
}
if (!getcwd(gitdir, sizeof(gitdir)-1))
die("Unable to read current working directory");
if (chdir(cwd))
die("Cannot come back to cwd");
/*
/*
* In case there is a work tree we may change the directory,
* Test in the following order (relative to the cwd):
* therefore make GIT_DIR an absolute path.
* - .git/
* - ./ (bare)
* - ../.git/
* - ../ (bare)
* - ../../.git/
* etc.
*/
*/
if (gitdirenv[0] != '/') {
offset = len = strlen(cwd);
setenv(GIT_DIR_ENVIRONMENT, gitdir, 1);
for (;;) {
gitdirenv = getenv(GIT_DIR_ENVIRONMENT);
if (is_git_directory(DEFAULT_GIT_DIR_ENVIRONMENT))
if (!gitdirenv)
break;
die("getenv after setenv failed");
if (is_git_directory(".")) {
if (PATH_MAX - 40 < strlen(gitdirenv)) {
inside_git_dir = 1;
if (nongit_ok) {
if (!work_tree_env)
*nongit_ok = 1;
inside_work_tree = 0;
return NULL;
setenv(GIT_DIR_ENVIRONMENT, ".", 1);
}
die("$%s too big after expansion to absolute path",
GIT_DIR_ENVIRONMENT);
}
}
strcat(cwd, "/");
strcat(gitdir, "/");
inside_git_dir = !prefixcmp(cwd, gitdir);
gitworktree = getenv(GIT_WORK_TREE_ENVIRONMENT);
if (!gitworktree) {
gitworktree_config = worktree;
worktree[0] = '\0';
}
git_config(git_setup_config);
if (!gitworktree) {
gitworktree_config = NULL;
if (worktree[0])
gitworktree = worktree;
if (gitworktree && gitworktree[0] != '/')
wt_rel_gitdir = 1;
}
if (wt_rel_gitdir && chdir(gitdirenv))
die("Cannot change directory to $%s '%s'",
GIT_DIR_ENVIRONMENT, gitdirenv);
if (gitworktree && chdir(gitworktree)) {
if (nongit_ok) {
if (wt_rel_gitdir && chdir(cwd))
die("Cannot come back to cwd");
*nongit_ok = 1;
return NULL;
return NULL;
}
}
if (wt_rel_gitdir)
chdir("..");
die("Cannot change directory to working tree '%s'"
do {
" from $%s", gitworktree, GIT_DIR_ENVIRONMENT);
if (!offset) {
else
if (nongit_ok) {
die("Cannot change directory to working tree '%s'",
if (chdir(cwd))
gitworktree);
die("Cannot come back to cwd");
}
*nongit_ok = 1;
if (!getcwd(worktree, sizeof(worktree)-1))
return NULL;
die("Unable to read current working directory");
}
strcat(worktree, "/");
die("Not a git repository");
inside_work_tree = !prefixcmp(cwd, worktree);
}
} while (cwd[--offset] != '/');
if (gitworktree && inside_work_tree && !prefixcmp(worktree, gitdir) &&
strcmp(worktree, gitdir)) {
inside_git_dir = 0;
}
}
if (!inside_work_tree) {
inside_git_dir = 0;
if (chdir(cwd))
if (!work_tree_env)
die("Cannot come back to cwd");
inside_work_tree = 1;
git_work_tree_cfg = xstrndup(cwd, offset);
if (offset == len)
return NULL;
return NULL;
}
if (!strcmp(cwd, worktree))
/* Make "offset" point to past the '/', and add a '/' at the end */
return NULL;
offset++;
return cwd+strlen(worktree);
cwd[len++] = '/';
cwd[len] = 0;
return cwd + offset;
}
}
int git_config_perm(const char *var, const char *value)
int git_config_perm(const char *var, const char *value)
@ -386,6 +340,16 @@ int check_repository_format_version(const char *var, const char *value)
repository_format_version = git_config_int(var, value);
repository_format_version = git_config_int(var, value);
else if (strcmp(var, "core.sharedrepository") == 0)
else if (strcmp(var, "core.sharedrepository") == 0)
shared_repository = git_config_perm(var, value);
shared_repository = git_config_perm(var, value);
else if (strcmp(var, "core.bare") == 0) {
is_bare_repository_cfg = git_config_bool(var, value);
if (is_bare_repository_cfg == 1)
inside_work_tree = -1;
} else if (strcmp(var, "core.worktree") == 0) {
if (git_work_tree_cfg)
free(git_work_tree_cfg);
git_work_tree_cfg = xstrdup(value);
inside_work_tree = -1;
}
return 0;
return 0;
}
}
@ -402,5 +366,16 @@ const char *setup_git_directory(void)
{
{
const char *retval = setup_git_directory_gently(NULL);
const char *retval = setup_git_directory_gently(NULL);
check_repository_format();
check_repository_format();
/* If the work tree is not the default one, recompute prefix */
if (inside_work_tree < 0) {
static char buffer[PATH_MAX + 1];
char *rel;
if (retval && chdir(retval))
die ("Could not jump back into original cwd");
rel = get_relative_cwd(buffer, PATH_MAX, get_git_work_tree());
return rel && *rel ? strcat(rel, "/") : NULL;
}
return retval;
return retval;
}
}