properly keep track of current working directory

Various failure modes in the repository detection code path currently
quote the wrong directory in their error message. The working directory
is changed iteratively to the parent directory until a git repository is
found. If the working directory cannot be changed to the parent
directory for some reason, the detection gives up and prints an error
message. The error message should report the current working directory.

Instead of continually updating the 'cwd' variable, which is actually
used to remember the original working directory, the 'offset' variable
is used to keep track of the current working directory. At the point
where the affected error handling code is called, 'offset' already
points to the end of the parent of the working directory, rather than
the current working directory.

Fix this by explicitly using a variable 'offset_parent' and update
'offset' concurrently with the call to chdir.

In a similar fashion, the function get_device_or_die() would print the
original working directory in case of a failure, rather than the current
working directory. Fix this as well by making use of the 'offset'
variable.

Lastly, replace the phrase 'mount parent' with 'mount point'. The former
appears to be a typo.

Signed-off-by: Clemens Buchacher <drizzd@aon.at>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Clemens Buchacher 2012-04-13 01:11:36 +02:00 committed by Junio C Hamano
parent e8dde3e5f9
commit 2565b43bd2

22
setup.c
View File

@ -569,13 +569,15 @@ static const char *setup_nongit(const char *cwd, int *nongit_ok)
return NULL; return NULL;
} }
static dev_t get_device_or_die(const char *path, const char *prefix) static dev_t get_device_or_die(const char *path, const char *prefix, int prefix_len)
{ {
struct stat buf; struct stat buf;
if (stat(path, &buf)) if (stat(path, &buf)) {
die_errno("failed to stat '%s%s%s'", die_errno("failed to stat '%*s%s%s'",
prefix_len,
prefix ? prefix : "", prefix ? prefix : "",
prefix ? "/" : "", path); prefix ? "/" : "", path);
}
return buf.st_dev; return buf.st_dev;
} }
@ -589,7 +591,7 @@ static const char *setup_git_directory_gently_1(int *nongit_ok)
static char cwd[PATH_MAX+1]; static char cwd[PATH_MAX+1];
const char *gitdirenv, *ret; const char *gitdirenv, *ret;
char *gitfile; char *gitfile;
int len, offset, ceil_offset; int len, offset, offset_parent, ceil_offset;
dev_t current_device = 0; dev_t current_device = 0;
int one_filesystem = 1; int one_filesystem = 1;
@ -631,7 +633,7 @@ static const char *setup_git_directory_gently_1(int *nongit_ok)
*/ */
one_filesystem = !git_env_bool("GIT_DISCOVERY_ACROSS_FILESYSTEM", 0); one_filesystem = !git_env_bool("GIT_DISCOVERY_ACROSS_FILESYSTEM", 0);
if (one_filesystem) if (one_filesystem)
current_device = get_device_or_die(".", NULL); current_device = get_device_or_die(".", NULL, 0);
for (;;) { for (;;) {
gitfile = (char*)read_gitfile(DEFAULT_GIT_DIR_ENVIRONMENT); gitfile = (char*)read_gitfile(DEFAULT_GIT_DIR_ENVIRONMENT);
if (gitfile) if (gitfile)
@ -653,11 +655,12 @@ static const char *setup_git_directory_gently_1(int *nongit_ok)
if (is_git_directory(".")) if (is_git_directory("."))
return setup_bare_git_dir(cwd, offset, len, nongit_ok); return setup_bare_git_dir(cwd, offset, len, nongit_ok);
while (--offset > ceil_offset && cwd[offset] != '/'); offset_parent = offset;
if (offset <= ceil_offset) while (--offset_parent > ceil_offset && cwd[offset_parent] != '/');
if (offset_parent <= ceil_offset)
return setup_nongit(cwd, nongit_ok); return setup_nongit(cwd, nongit_ok);
if (one_filesystem) { if (one_filesystem) {
dev_t parent_device = get_device_or_die("..", cwd); dev_t parent_device = get_device_or_die("..", cwd, offset);
if (parent_device != current_device) { if (parent_device != current_device) {
if (nongit_ok) { if (nongit_ok) {
if (chdir(cwd)) if (chdir(cwd))
@ -666,7 +669,7 @@ static const char *setup_git_directory_gently_1(int *nongit_ok)
return NULL; return NULL;
} }
cwd[offset] = '\0'; cwd[offset] = '\0';
die("Not a git repository (or any parent up to mount parent %s)\n" die("Not a git repository (or any parent up to mount point %s)\n"
"Stopping at filesystem boundary (GIT_DISCOVERY_ACROSS_FILESYSTEM not set).", cwd); "Stopping at filesystem boundary (GIT_DISCOVERY_ACROSS_FILESYSTEM not set).", cwd);
} }
} }
@ -674,6 +677,7 @@ static const char *setup_git_directory_gently_1(int *nongit_ok)
cwd[offset] = '\0'; cwd[offset] = '\0';
die_errno("Cannot change to '%s/..'", cwd); die_errno("Cannot change to '%s/..'", cwd);
} }
offset = offset_parent;
} }
} }