2 * mono-path.c: Routines for handling path names.
5 * Gonzalo Paniagua Javier (gonzalo@novell.com)
6 * Miguel de Icaza (miguel@novell.com)
8 * (C) 2006 Novell, Inc. http://www.novell.com
19 /* This is only needed for the mono_path_canonicalize code, MAXSYMLINKS, could be moved */
20 #ifdef HAVE_SYS_PARAM_H
21 #include <sys/param.h>
24 #include "mono-path.h"
26 /* Resolves '..' and '.' references in a path. If the path provided is relative,
27 * it will be relative to the current directory */
29 mono_path_canonicalize (const char *path)
31 gchar *abspath, *pos, *lastpos, *dest;
34 if (g_path_is_absolute (path)) {
35 abspath = g_strdup (path);
37 gchar *tmpdir = g_get_current_dir ();
38 abspath = g_build_filename (tmpdir, path, NULL);
43 g_strdelimit (abspath, "/", '\\');
45 abspath = g_strreverse (abspath);
48 dest = lastpos = abspath;
49 pos = strchr (lastpos, G_DIR_SEPARATOR);
52 int len = pos - lastpos;
53 if (len == 1 && lastpos [0] == '.') {
55 } else if (len == 2 && lastpos [0] == '.' && lastpos [1] == '.') {
62 /* The two strings can overlap */
63 memmove (dest, lastpos, len + 1);
68 pos = strchr (lastpos, G_DIR_SEPARATOR);
71 #ifdef PLATFORM_WIN32 /* For UNC paths the first '\' is removed. */
72 if (*(lastpos-1) == G_DIR_SEPARATOR && *(lastpos-2) == G_DIR_SEPARATOR)
76 if (dest != lastpos) strcpy (dest, lastpos);
77 return g_strreverse (abspath);
81 * This ensures that the path that we store points to the final file
82 * not a path to a symlink.
85 mono_path_resolve_symlinks (const char *path)
88 return mono_path_canonicalize (path);
90 char *p, *concat, *dir;
91 char buffer [PATH_MAX+1];
92 int n, iterations = 0;
97 n = readlink (p, buffer, sizeof (buffer)-1);
100 p = mono_path_canonicalize (copy);
106 if (!g_path_is_absolute (buffer)) {
107 dir = g_path_get_dirname (p);
108 concat = g_build_filename (dir, buffer, NULL);
111 concat = g_strdup (buffer);
114 p = mono_path_canonicalize (concat);
116 } while (iterations < MAXSYMLINKS);