abspath: convert real_path_internal() to strbuf
Use strbuf instead of fixed-sized buffers in real_path() in order to avoid the size limitations of the latter. Signed-off-by: Rene Scharfe <l.s.r@web.de> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:

committed by
Junio C Hamano

parent
251277acdf
commit
2fdb9ce067
69
abspath.c
69
abspath.c
@ -33,7 +33,7 @@ int is_directory(const char *path)
|
|||||||
*/
|
*/
|
||||||
static const char *real_path_internal(const char *path, int die_on_error)
|
static const char *real_path_internal(const char *path, int die_on_error)
|
||||||
{
|
{
|
||||||
static char bufs[2][PATH_MAX + 1], *buf = bufs[0], *next_buf = bufs[1];
|
static struct strbuf sb = STRBUF_INIT;
|
||||||
char *retval = NULL;
|
char *retval = NULL;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -43,14 +43,12 @@ static const char *real_path_internal(const char *path, int die_on_error)
|
|||||||
*/
|
*/
|
||||||
struct strbuf cwd = STRBUF_INIT;
|
struct strbuf cwd = STRBUF_INIT;
|
||||||
|
|
||||||
int buf_index = 1;
|
|
||||||
|
|
||||||
int depth = MAXDEPTH;
|
int depth = MAXDEPTH;
|
||||||
char *last_elem = NULL;
|
char *last_elem = NULL;
|
||||||
struct stat st;
|
struct stat st;
|
||||||
|
|
||||||
/* We've already done it */
|
/* We've already done it */
|
||||||
if (path == buf || path == next_buf)
|
if (path == sb.buf)
|
||||||
return path;
|
return path;
|
||||||
|
|
||||||
if (!*path) {
|
if (!*path) {
|
||||||
@ -60,26 +58,22 @@ static const char *real_path_internal(const char *path, int die_on_error)
|
|||||||
goto error_out;
|
goto error_out;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (strlcpy(buf, path, PATH_MAX) >= PATH_MAX) {
|
strbuf_reset(&sb);
|
||||||
if (die_on_error)
|
strbuf_addstr(&sb, path);
|
||||||
die("Too long path: %.*s", 60, path);
|
|
||||||
else
|
|
||||||
goto error_out;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (depth--) {
|
while (depth--) {
|
||||||
if (!is_directory(buf)) {
|
if (!is_directory(sb.buf)) {
|
||||||
char *last_slash = find_last_dir_sep(buf);
|
char *last_slash = find_last_dir_sep(sb.buf);
|
||||||
if (last_slash) {
|
if (last_slash) {
|
||||||
last_elem = xstrdup(last_slash + 1);
|
last_elem = xstrdup(last_slash + 1);
|
||||||
last_slash[1] = '\0';
|
strbuf_setlen(&sb, last_slash - sb.buf + 1);
|
||||||
} else {
|
} else {
|
||||||
last_elem = xstrdup(buf);
|
last_elem = xmemdupz(sb.buf, sb.len);
|
||||||
*buf = '\0';
|
strbuf_reset(&sb);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (*buf) {
|
if (sb.len) {
|
||||||
if (!cwd.len && strbuf_getcwd(&cwd)) {
|
if (!cwd.len && strbuf_getcwd(&cwd)) {
|
||||||
if (die_on_error)
|
if (die_on_error)
|
||||||
die_errno("Could not get current working directory");
|
die_errno("Could not get current working directory");
|
||||||
@ -87,14 +81,15 @@ static const char *real_path_internal(const char *path, int die_on_error)
|
|||||||
goto error_out;
|
goto error_out;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (chdir(buf)) {
|
if (chdir(sb.buf)) {
|
||||||
if (die_on_error)
|
if (die_on_error)
|
||||||
die_errno("Could not switch to '%s'", buf);
|
die_errno("Could not switch to '%s'",
|
||||||
|
sb.buf);
|
||||||
else
|
else
|
||||||
goto error_out;
|
goto error_out;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!getcwd(buf, PATH_MAX)) {
|
if (strbuf_getcwd(&sb)) {
|
||||||
if (die_on_error)
|
if (die_on_error)
|
||||||
die_errno("Could not get current working directory");
|
die_errno("Could not get current working directory");
|
||||||
else
|
else
|
||||||
@ -102,44 +97,30 @@ static const char *real_path_internal(const char *path, int die_on_error)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (last_elem) {
|
if (last_elem) {
|
||||||
size_t len = strlen(buf);
|
if (sb.len && !is_dir_sep(sb.buf[sb.len - 1]))
|
||||||
if (len + strlen(last_elem) + 2 > PATH_MAX) {
|
strbuf_addch(&sb, '/');
|
||||||
if (die_on_error)
|
strbuf_addstr(&sb, last_elem);
|
||||||
die("Too long path name: '%s/%s'",
|
|
||||||
buf, last_elem);
|
|
||||||
else
|
|
||||||
goto error_out;
|
|
||||||
}
|
|
||||||
if (len && !is_dir_sep(buf[len - 1]))
|
|
||||||
buf[len++] = '/';
|
|
||||||
strcpy(buf + len, last_elem);
|
|
||||||
free(last_elem);
|
free(last_elem);
|
||||||
last_elem = NULL;
|
last_elem = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!lstat(buf, &st) && S_ISLNK(st.st_mode)) {
|
if (!lstat(sb.buf, &st) && S_ISLNK(st.st_mode)) {
|
||||||
ssize_t len = readlink(buf, next_buf, PATH_MAX);
|
struct strbuf next_sb = STRBUF_INIT;
|
||||||
|
ssize_t len = strbuf_readlink(&next_sb, sb.buf, 0);
|
||||||
if (len < 0) {
|
if (len < 0) {
|
||||||
if (die_on_error)
|
if (die_on_error)
|
||||||
die_errno("Invalid symlink '%s'", buf);
|
die_errno("Invalid symlink '%s'",
|
||||||
|
sb.buf);
|
||||||
else
|
else
|
||||||
goto error_out;
|
goto error_out;
|
||||||
}
|
}
|
||||||
if (PATH_MAX <= len) {
|
strbuf_swap(&sb, &next_sb);
|
||||||
if (die_on_error)
|
strbuf_release(&next_sb);
|
||||||
die("symbolic link too long: %s", buf);
|
|
||||||
else
|
|
||||||
goto error_out;
|
|
||||||
}
|
|
||||||
next_buf[len] = '\0';
|
|
||||||
buf = next_buf;
|
|
||||||
buf_index = 1 - buf_index;
|
|
||||||
next_buf = bufs[buf_index];
|
|
||||||
} else
|
} else
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
retval = buf;
|
retval = sb.buf;
|
||||||
error_out:
|
error_out:
|
||||||
free(last_elem);
|
free(last_elem);
|
||||||
if (cwd.len && chdir(cwd.buf))
|
if (cwd.len && chdir(cwd.buf))
|
||||||
|
Reference in New Issue
Block a user