2001-08-10 15:05:27 +00:00
|
|
|
/* vi: set sw=4 ts=4: */
|
|
|
|
/*
|
2003-03-19 09:13:01 +00:00
|
|
|
* bb_simplify_path implementation for busybox
|
2001-08-10 15:05:27 +00:00
|
|
|
*
|
2003-03-19 09:13:01 +00:00
|
|
|
* Copyright (C) 2001 Manuel Novoa III <mjn3@codepoet.org>
|
2001-08-10 15:05:27 +00:00
|
|
|
*
|
2006-07-10 11:41:19 +00:00
|
|
|
* Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
|
2001-08-10 15:05:27 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "libbb.h"
|
|
|
|
|
2003-03-19 09:13:01 +00:00
|
|
|
char *bb_simplify_path(const char *path)
|
2001-08-10 15:05:27 +00:00
|
|
|
{
|
2001-08-14 17:10:08 +00:00
|
|
|
char *s, *start, *p;
|
2001-08-10 15:05:27 +00:00
|
|
|
|
|
|
|
if (path[0] == '/')
|
2006-08-03 15:41:12 +00:00
|
|
|
start = xstrdup(path);
|
2001-08-10 15:05:27 +00:00
|
|
|
else {
|
2007-02-11 16:19:28 +00:00
|
|
|
s = xrealloc_getcwd_or_warn(NULL);
|
2001-08-14 17:10:08 +00:00
|
|
|
start = concat_path_file(s, path);
|
|
|
|
free(s);
|
2001-08-10 15:05:27 +00:00
|
|
|
}
|
2001-08-14 17:10:08 +00:00
|
|
|
p = s = start;
|
2001-08-10 15:05:27 +00:00
|
|
|
|
2001-08-14 17:10:08 +00:00
|
|
|
do {
|
|
|
|
if (*p == '/') {
|
|
|
|
if (*s == '/') { /* skip duplicate (or initial) slash */
|
|
|
|
continue;
|
|
|
|
} else if (*s == '.') {
|
|
|
|
if (s[1] == '/' || s[1] == 0) { /* remove extra '.' */
|
|
|
|
continue;
|
|
|
|
} else if ((s[1] == '.') && (s[2] == '/' || s[2] == 0)) {
|
|
|
|
++s;
|
|
|
|
if (p > start) {
|
|
|
|
while (*--p != '/'); /* omit previous dir */
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2001-08-10 15:05:27 +00:00
|
|
|
}
|
|
|
|
}
|
2001-08-14 17:10:08 +00:00
|
|
|
*++p = *s;
|
|
|
|
} while (*++s);
|
|
|
|
|
|
|
|
if ((p == start) || (*p != '/')) { /* not a trailing slash */
|
|
|
|
++p; /* so keep last character */
|
2001-08-10 15:05:27 +00:00
|
|
|
}
|
2001-08-14 17:10:08 +00:00
|
|
|
*p = 0;
|
|
|
|
|
2001-08-10 15:05:27 +00:00
|
|
|
return start;
|
|
|
|
}
|