From f6f23ad2d547734fadc42b7410722931f920070d Mon Sep 17 00:00:00 2001 From: Alan Modra Date: Thu, 18 Jul 2002 06:44:35 +0000 Subject: [PATCH] prefix.c: (update_path): Strip ".." components when prior dir doesn't exist. * prefix.c: (update_path): Strip ".." components when prior dir doesn't exist. Pass correct var to UPDATE_PATH_HOST_CANONICALIZE. From-SVN: r55547 --- gcc/ChangeLog | 3 +++ gcc/prefix.c | 68 ++++++++++++++++++++++++++++++++++++++++++++++++--- 2 files changed, 68 insertions(+), 3 deletions(-) diff --git a/gcc/ChangeLog b/gcc/ChangeLog index d18e92c087c..f99c294f9c1 100644 --- a/gcc/ChangeLog +++ b/gcc/ChangeLog @@ -1,5 +1,8 @@ 2002-07-18 Alan Modra + * prefix.c: (update_path): Strip ".." components when prior dir + doesn't exist. Pass correct var to UPDATE_PATH_HOST_CANONICALIZE. + * config/rs6000/sysv4.h (ASM_OUTPUT_REG_PUSH): Remove 64-bit support. (ASM_OUTPUT_REG_POP): Likewise. diff --git a/gcc/prefix.c b/gcc/prefix.c index b7b162aac49..c8f0b98feb4 100644 --- a/gcc/prefix.c +++ b/gcc/prefix.c @@ -1,5 +1,6 @@ /* Utility to update paths from internal to external forms. - Copyright (C) 1997, 1998, 1999, 2000, 2001 Free Software Foundation, Inc. + Copyright (C) 1997, 1998, 1999, 2000, 2001, 2002 + Free Software Foundation, Inc. This file is part of GCC. @@ -251,7 +252,7 @@ update_path (path, key) const char *path; const char *key; { - char *result; + char *result, *p; if (! strncmp (path, std_prefix, strlen (std_prefix)) && key != 0) { @@ -271,9 +272,70 @@ update_path (path, key) else result = xstrdup (path); +#ifndef ALWAYS_STRIP_DOTDOT +#define ALWAYS_STRIP_DOTDOT 0 +#endif + + p = result; + while (1) + { + char *src, *dest; + + p = strchr (p, '.'); + if (p == NULL) + break; + /* Get rid of a leading `./' and replace `/./' with `/'. */ + if (IS_DIR_SEPARATOR (p[1]) + && (p == result || IS_DIR_SEPARATOR (p[-1]))) + { + src = p + 2; + /* Be careful about .//foo */ + while (IS_DIR_SEPARATOR (*src)) + ++src; + dest = p; + while ((*dest++ = *src++) != 0) + ; + } + /* Look for `/../' */ + else if (p[1] == '.' + && IS_DIR_SEPARATOR (p[2]) + && (p != result && IS_DIR_SEPARATOR (p[-1]))) + { + *p = 0; + if (!ALWAYS_STRIP_DOTDOT && access (result, X_OK) == 0) + { + *p = '.'; + p += 3; + } + else + { + /* We can't access the dir, so we won't be able to + access dir/.. either. Strip out dir/.. We know dir + isn't `.' because we've rid ourselves of `.' path + components above. */ + dest = p - 1; + while (dest != result && IS_DIR_SEPARATOR (*dest)) + --dest; + while (dest != result && IS_DIR_SEPARATOR (dest[-1])) + --dest; + /* Don't strip leading `/'. */ + while (IS_DIR_SEPARATOR (*dest)) + ++dest; + src = p + 3; + while (IS_DIR_SEPARATOR (*src)) + ++src; + p = dest; + while ((*dest++ = *src++) != 0) + ; + } + } + else + ++p; + } + #ifdef UPDATE_PATH_HOST_CANONICALIZE /* Perform host dependent canonicalization when needed. */ - UPDATE_PATH_HOST_CANONICALIZE (path); + UPDATE_PATH_HOST_CANONICALIZE (result); #endif #ifdef DIR_SEPARATOR_2