annotate src/realpath.c @ 5781:0853e1ec8529

Use alloca_{rawbytes,ibytes} in #'copy-file, #'insert-file-contents-internal src/ChangeLog addition: 2014-01-20 Aidan Kehoe <kehoea@parhasard.net> * fileio.c (Fcopy_file, Finsert_file_contents_internal): Use alloca_{rawbytes,ibytes} here instead of the implicit alloca on the stack; doesn't change where the buffers are allocated for these two functions, but does mean that decisions about alloca vs. malloc based on buffer size are made in the same place (ultimately, the ALLOCA() macro).
author Aidan Kehoe <kehoea@parhasard.net>
date Mon, 20 Jan 2014 17:53:07 +0000
parents 308d34e9f07d
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
1 /*
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
2 * realpath.c -- canonicalize pathname by removing symlinks
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
3 * Copyright (C) 1993 Rick Sladkey <jrs@world.std.com>
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
4 * Copyright (C) 2001, 2002, 2004 Ben Wing.
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
5 *
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
6
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
7 This file is part of XEmacs.
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
8
5402
308d34e9f07d Changed bulk of GPLv2 or later files identified by script
Mats Lidell <matsl@xemacs.org>
parents: 4982
diff changeset
9 XEmacs is free software: you can redistribute it and/or modify it
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
10 under the terms of the GNU General Public License as published by the
5402
308d34e9f07d Changed bulk of GPLv2 or later files identified by script
Mats Lidell <matsl@xemacs.org>
parents: 4982
diff changeset
11 Free Software Foundation, either version 3 of the License, or (at your
308d34e9f07d Changed bulk of GPLv2 or later files identified by script
Mats Lidell <matsl@xemacs.org>
parents: 4982
diff changeset
12 option) any later version.
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
13
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
14 XEmacs is distributed in the hope that it will be useful, but WITHOUT
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
15 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
16 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
17 for more details.
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
18
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
19 You should have received a copy of the GNU General Public License
5402
308d34e9f07d Changed bulk of GPLv2 or later files identified by script
Mats Lidell <matsl@xemacs.org>
parents: 4982
diff changeset
20 along with XEmacs. If not, see <http://www.gnu.org/licenses/>. */
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
21
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
22 /* Synched up with: Not in FSF. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
23
771
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
24 /* This file has been Mule-ized, June 2001 by Ben Wing.
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
25
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
26 Everything in this file now works in terms of internal, not external,
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
27 data. This is the only way to be safe, and it makes the code cleaner. */
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
28
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
29 #include <config.h>
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
30 #include "lisp.h"
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
31
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
32 #include "profile.h"
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
33
558
ed498ef2108b [xemacs-hg @ 2001-05-23 09:59:33 by ben]
ben
parents: 462
diff changeset
34 #include "sysfile.h"
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
35 #include "sysdir.h"
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
36
771
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
37 #define MAX_READLINKS 32
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
38
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
39 #ifdef WIN32_ANY
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
40 #include "syswindows.h"
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
41 #ifndef ELOOP
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
42 #define ELOOP 10062 /* = WSAELOOP in winsock.h */
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
43 #endif
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
44 #endif
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
45
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
46 Lisp_Object QSin_qxe_realpath;
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
47
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
48 /* Length of start of absolute filename. */
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
49 static int
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
50 abs_start (const Ibyte *name)
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
51 {
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
52 #ifdef WIN32_ANY
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
53 if (isalpha (*name) && IS_DEVICE_SEP (name[1])
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
54 && IS_DIRECTORY_SEP (name[2]))
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
55 return 3;
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
56 else if (IS_DIRECTORY_SEP (*name))
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
57 return IS_DIRECTORY_SEP (name[1]) ? 2 : 1;
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
58 else
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
59 return 0;
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
60 #else /* not WIN32_ANY */
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
61 return IS_DIRECTORY_SEP (*name) ? 1 : 0;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
62 #endif
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
63 }
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
64
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
65 /* Find real name of a file by resolving symbolic links and/or shortcuts
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
66 under Windows (.LNK links), if such support is enabled.
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
67
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
68 If no link found, and LINKS_ONLY is false, look up the correct case in
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
69 the file system of the last component.
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
70
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
71 Under Windows, UNC servers and shares are lower-cased. Directories must
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
72 be given without trailing '/'. One day, this could read Win2K's reparse
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
73 points.
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
74
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
75 Returns length of characters copied info BUF.
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
76 DOES NOT ZERO TERMINATE!!!!!
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
77 */
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
78
4721
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
79 #ifdef REALPATH_CORRECTS_CASE /* Darwin */
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
80 #include <sys/param.h>
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
81 #include <stdlib.h>
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
82 #endif
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
83
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
84 static int
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
85 readlink_or_correct_case (const Ibyte *name, Ibyte *buf, Bytecount size,
3042
967fea9dfad5 [xemacs-hg @ 2005-11-02 03:30:17 by crestani]
crestani
parents: 2526
diff changeset
86 #ifndef WIN32_ANY
3044
d00888bfced1 [xemacs-hg @ 2005-11-02 10:01:57 by crestani]
crestani
parents: 3042
diff changeset
87 Boolint UNUSED (links_only)
3042
967fea9dfad5 [xemacs-hg @ 2005-11-02 03:30:17 by crestani]
crestani
parents: 2526
diff changeset
88 #else
3044
d00888bfced1 [xemacs-hg @ 2005-11-02 10:01:57 by crestani]
crestani
parents: 3042
diff changeset
89 Boolint links_only
3042
967fea9dfad5 [xemacs-hg @ 2005-11-02 03:30:17 by crestani]
crestani
parents: 2526
diff changeset
90 #endif
3044
d00888bfced1 [xemacs-hg @ 2005-11-02 10:01:57 by crestani]
crestani
parents: 3042
diff changeset
91 )
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
92 {
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
93 #ifndef WIN32_ANY
4721
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
94 #ifdef REALPATH_CORRECTS_CASE
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
95 /* Darwin's realpath corrects file name case, so we want to use that
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
96 here, as well as our own, non-case-correcting, implementation
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
97 further down in this file.
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
98
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
99 It might be reasonable to incorporate case correction in our own
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
100 realpath implementation, which would help things with
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
101 case-insensitive file systems on Linux; one way to do this would
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
102 be to make sure that init_initial_directory and
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
103 get_initial_directory always give the correct case. */
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
104 int n = qxe_readlink (name, buf, (size_t) size);
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
105 Extbyte realpath_buf[PATH_MAX], *tmp;
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
106 DECLARE_EISTRING (realpathing);
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
107
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
108 if (n >= 0 || errno != EINVAL)
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
109 return n;
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
110
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
111 eicpy_rawz (realpathing, name);
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
112 eito_external (realpathing, Qfile_name);
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
113 tmp = realpath (eiextdata (realpathing), realpath_buf);
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
114
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
115 if (!tmp)
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
116 return -1;
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
117
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
118 if (0 == memcmp (eiextdata (realpathing), realpath_buf,
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
119 eiextlen (realpathing)))
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
120 {
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
121 /* No case change needed; tell the caller that. */
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
122 errno = EINVAL;
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
123 return -1;
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
124 }
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
125
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
126 eireset (realpathing);
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
127 eicpy_ext (realpathing, realpath_buf, Qfile_name);
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
128 if (eilen (realpathing) > size)
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
129 {
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
130 errno = ERANGE;
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
131 return -1;
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
132 }
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
133
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
134 memcpy (buf, eidata (realpathing), eilen (realpathing));
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
135 return eilen (realpathing);
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
136 #else /* !REALPATH_CORRECTS_CASE */
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
137 return qxe_readlink (name, buf, (size_t) size);
4721
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
138 #endif /* REALPATH_CORRECTS_CASE */
19d70297d866 Make readlink_or_correct_case function correctly on Darwin.
Aidan Kehoe <kehoea@parhasard.net>
parents: 3044
diff changeset
139 #else /* defined (WIN32_ANY) */
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
140 # ifdef CYGWIN
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
141 int n = qxe_readlink (name, buf, (size_t) size);
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
142 if (n >= 0 || errno != EINVAL)
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
143 return n;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
144
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
145 /* The file may exist, but isn't a symlink. Try to find the
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
146 right name. */
4834
b3ea9c582280 Use new cygwin_conv_path API with Cygwin 1.7 for converting names between Win32 and POSIX, UTF-8-aware, with attendant changes elsewhere
Ben Wing <ben@xemacs.org>
parents: 4721
diff changeset
147 LOCAL_FILE_FORMAT_TO_INTERNAL_MSWIN (name, name);
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
148 # else
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
149 if (mswindows_shortcuts_are_symlinks)
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
150 {
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
151 Ibyte *tmp = mswindows_read_link (name);
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
152
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
153 if (tmp != NULL)
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
154 {
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
155 /* Fucking fixed buffers. */
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
156 Bytecount len = qxestrlen (tmp);
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
157 if (len > size)
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
158 {
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
159 errno = ENAMETOOLONG;
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
160 return -1;
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
161 }
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
162 memcpy (buf, tmp, len);
4976
16112448d484 Rename xfree(FOO, TYPE) -> xfree(FOO)
Ben Wing <ben@xemacs.org>
parents: 4952
diff changeset
163 xfree (tmp);
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
164 return len;
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
165 }
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
166 }
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
167 # endif
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
168
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
169 if (links_only)
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
170 {
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
171 errno = EINVAL;
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
172 return -1;
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
173 }
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
174
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
175 {
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
176 int len = 0;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
177 int err = 0;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
178 const Ibyte *lastname;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
179 int count = 0;
1204
e22b0213b713 [xemacs-hg @ 2003-01-12 11:07:58 by michaels]
michaels
parents: 1116
diff changeset
180 const Ibyte *nn;
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
181 DECLARE_EISTRING (result);
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
182
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
183 assert (*name);
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
184
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
185 /* Sort of check we have a valid filename. */
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
186 if (qxestrpbrk (name, "*?|<>\""))
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
187 {
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
188 errno = ENOENT;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
189 return -1;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
190 }
2421
ab71ad6ff3dd [xemacs-hg @ 2004-12-06 03:50:53 by ben]
ben
parents: 2367
diff changeset
191 else if (qxestrlen (name) >= PATH_MAX_INTERNAL)
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
192 {
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
193 errno = ENAMETOOLONG;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
194 return -1;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
195 }
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
196
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
197 /* Find start of filename */
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
198 lastname = name + qxestrlen (name);
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
199 while (lastname > name && !IS_DIRECTORY_SEP (lastname[-1]))
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
200 --lastname;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
201
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
202 /* Count slashes in unc path */
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
203 if (abs_start (name) == 2)
1204
e22b0213b713 [xemacs-hg @ 2003-01-12 11:07:58 by michaels]
michaels
parents: 1116
diff changeset
204 for (nn = name; *nn; nn++)
e22b0213b713 [xemacs-hg @ 2003-01-12 11:07:58 by michaels]
michaels
parents: 1116
diff changeset
205 if (IS_DIRECTORY_SEP (*nn))
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
206 count++;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
207
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
208 if (count >= 2 && count < 4)
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
209 {
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
210 eicpy_rawz (result, lastname);
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
211 eilwr (result);
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
212 }
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
213 else
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
214 {
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
215 WIN32_FIND_DATAW find_data;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
216 Extbyte *nameext;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
217 HANDLE dir_handle;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
218
4981
4aebb0131297 Cleanups/renaming of EXTERNAL_TO_C_STRING and friends
Ben Wing <ben@xemacs.org>
parents: 4952
diff changeset
219 nameext = ITEXT_TO_TSTR (name);
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
220 dir_handle = qxeFindFirstFile (nameext, &find_data);
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
221 if (dir_handle == INVALID_HANDLE_VALUE)
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
222 {
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
223 errno = ENOENT;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
224 return -1;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
225 }
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
226 eicpy_ext (result, (Extbyte *) find_data.cFileName, Qmswindows_tstr);
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
227 FindClose (dir_handle);
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
228 }
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
229
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
230 if ((len = eilen (result)) <= size)
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
231 {
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
232 DECLARE_EISTRING (eilastname);
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
233
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
234 eicpy_rawz (eilastname, lastname);
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
235 if (eicmp_ei (eilastname, result) == 0)
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
236 /* Signal that the name is already OK. */
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
237 err = EINVAL;
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
238 else
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
239 memcpy (buf, eidata (result), len);
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
240 }
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
241 else
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
242 err = ENAMETOOLONG;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
243
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
244 errno = err;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
245 return err ? -1 : len;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
246 }
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
247 #endif /* WIN32_ANY */
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
248 }
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
249
771
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
250 /* Mule Note: This function works with and returns
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
251 internally-formatted strings.
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
252
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
253 if LINKS_ONLY is true, don't do case canonicalization under
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
254 Windows. */
440
8de8e3f6228a Import from CVS: tag r21-2-28
cvs
parents: 428
diff changeset
255
867
804517e16990 [xemacs-hg @ 2002-06-05 09:54:39 by ben]
ben
parents: 851
diff changeset
256 Ibyte *
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
257 qxe_realpath (const Ibyte *path, Ibyte *resolved_path, Boolint links_only)
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
258 {
2421
ab71ad6ff3dd [xemacs-hg @ 2004-12-06 03:50:53 by ben]
ben
parents: 2367
diff changeset
259 Ibyte copy_path[PATH_MAX_INTERNAL];
867
804517e16990 [xemacs-hg @ 2002-06-05 09:54:39 by ben]
ben
parents: 851
diff changeset
260 Ibyte *new_path = resolved_path;
804517e16990 [xemacs-hg @ 2002-06-05 09:54:39 by ben]
ben
parents: 851
diff changeset
261 Ibyte *max_path;
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
262 Ibyte *retval = NULL;
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
263 #if defined (HAVE_READLINK) || defined (WIN32_ANY)
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
264 int readlinks = 0;
2421
ab71ad6ff3dd [xemacs-hg @ 2004-12-06 03:50:53 by ben]
ben
parents: 2367
diff changeset
265 Ibyte link_path[PATH_MAX_INTERNAL];
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
266 int n;
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
267 int abslen = abs_start (path);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
268 #endif
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
269
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
270 PROFILE_DECLARE ();
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
271
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
272 PROFILE_RECORD_ENTERING_SECTION (QSin_qxe_realpath);
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
273
1760
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
274 restart:
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
275
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
276 /* Make a copy of the source path since we may need to modify it. */
771
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
277 qxestrcpy (copy_path, path);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
278 path = copy_path;
2421
ab71ad6ff3dd [xemacs-hg @ 2004-12-06 03:50:53 by ben]
ben
parents: 2367
diff changeset
279 max_path = copy_path + PATH_MAX_INTERNAL - 2;
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
280
819
6504113e7c2d [xemacs-hg @ 2002-04-25 18:03:23 by andyp]
andyp
parents: 771
diff changeset
281 if (0)
6504113e7c2d [xemacs-hg @ 2002-04-25 18:03:23 by andyp]
andyp
parents: 771
diff changeset
282 ;
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
283 #ifdef WIN32_ANY
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
284 /* Check for c:/... or //server/... */
988
5df795348f45 [xemacs-hg @ 2002-09-01 22:13:52 by andyp]
andyp
parents: 867
diff changeset
285 else if (abslen == 3 || abslen == 2)
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
286 {
462
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 446
diff changeset
287 /* Make sure drive letter is lowercased. */
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
288 if (abslen == 3)
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
289 {
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
290 *new_path = tolower (*path);
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
291 new_path++;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
292 path++;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
293 abslen--;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
294 }
988
5df795348f45 [xemacs-hg @ 2002-09-01 22:13:52 by andyp]
andyp
parents: 867
diff changeset
295 /* Coerce directory chars. */
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
296 while (abslen-- > 0)
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
297 {
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
298 if (IS_DIRECTORY_SEP (*path))
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
299 *new_path++ = DIRECTORY_SEP;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
300 else
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
301 *new_path++ = *path;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
302 path++;
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
303 }
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
304 }
819
6504113e7c2d [xemacs-hg @ 2002-04-25 18:03:23 by andyp]
andyp
parents: 771
diff changeset
305 #endif
6504113e7c2d [xemacs-hg @ 2002-04-25 18:03:23 by andyp]
andyp
parents: 771
diff changeset
306 #ifdef WIN32_NATIVE
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
307 /* No drive letter, but a beginning slash? Prepend drive letter. */
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
308 else if (abslen == 1)
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
309 {
2421
ab71ad6ff3dd [xemacs-hg @ 2004-12-06 03:50:53 by ben]
ben
parents: 2367
diff changeset
310 get_initial_directory (new_path, PATH_MAX_INTERNAL - 1);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
311 new_path += 3;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
312 path++;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
313 }
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
314 /* Just a path name, prepend the current directory */
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
315 else
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
316 {
2421
ab71ad6ff3dd [xemacs-hg @ 2004-12-06 03:50:53 by ben]
ben
parents: 2367
diff changeset
317 get_initial_directory (new_path, PATH_MAX_INTERNAL - 1);
771
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
318 new_path += qxestrlen (new_path);
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
319 if (!IS_DIRECTORY_SEP (new_path[-1]))
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
320 *new_path++ = DIRECTORY_SEP;
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
321 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
322 #else
771
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
323 /* If it's a relative pathname use get_initial_directory for starters. */
819
6504113e7c2d [xemacs-hg @ 2002-04-25 18:03:23 by andyp]
andyp
parents: 771
diff changeset
324 else if (abslen == 0)
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
325 {
2421
ab71ad6ff3dd [xemacs-hg @ 2004-12-06 03:50:53 by ben]
ben
parents: 2367
diff changeset
326 get_initial_directory (new_path, PATH_MAX_INTERNAL - 1);
771
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
327 new_path += qxestrlen (new_path);
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
328 if (!IS_DIRECTORY_SEP (new_path[-1]))
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
329 *new_path++ = DIRECTORY_SEP;
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
330 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
331 else
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
332 {
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
333 /* Copy first directory sep. May have two on cygwin. */
771
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
334 qxestrncpy (new_path, path, abslen);
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
335 new_path += abslen;
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
336 path += abslen;
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
337 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
338 #endif
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
339 /* Expand each slash-separated pathname component. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
340 while (*path != '\0')
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
341 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
342 /* Ignore stray "/". */
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
343 if (IS_DIRECTORY_SEP (*path))
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
344 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
345 path++;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
346 continue;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
347 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
348
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
349 if (*path == '.')
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
350 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
351 /* Ignore ".". */
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
352 if (path[1] == '\0' || IS_DIRECTORY_SEP (path[1]))
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
353 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
354 path++;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
355 continue;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
356 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
357
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
358 /* Handle ".." */
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
359 if (path[1] == '.' &&
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
360 (path[2] == '\0' || IS_DIRECTORY_SEP (path[2])))
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
361 {
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
362 path += 2;
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
363
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
364 /* Ignore ".." at root. */
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
365 if (new_path == resolved_path + abs_start (resolved_path))
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
366 continue;
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
367
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
368 /* Handle ".." by backing up. */
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
369 --new_path;
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
370 while (!IS_DIRECTORY_SEP (new_path[-1]))
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
371 --new_path;
442
abe6d1db359e Import from CVS: tag r21-2-36
cvs
parents: 440
diff changeset
372 continue;
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
373 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
374 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
375
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
376 /* Safely copy the next pathname component. */
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
377 while (*path != '\0' && !IS_DIRECTORY_SEP (*path))
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
378 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
379 if (path > max_path)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
380 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
381 errno = ENAMETOOLONG;
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
382 goto done;
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
383 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
384 *new_path++ = *path++;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
385 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
386
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
387 #if defined (HAVE_READLINK) || defined (WIN32_ANY)
771
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
388 /* See if latest pathname component is a symlink or needs case
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
389 correction. */
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
390 *new_path = '\0';
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
391 n = readlink_or_correct_case (resolved_path, link_path,
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
392 PATH_MAX_INTERNAL - 1, links_only);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
393
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
394 if (n < 0)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
395 {
771
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
396 /* EINVAL means the file exists but isn't a symlink or doesn't
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
397 need case correction. */
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
398 #ifdef WIN32_ANY
462
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 446
diff changeset
399 if (errno != EINVAL && errno != ENOENT)
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 446
diff changeset
400 #else
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 446
diff changeset
401 if (errno != EINVAL)
0784d089fdc9 Import from CVS: tag r21-2-46
cvs
parents: 446
diff changeset
402 #endif
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
403 goto done;
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
404 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
405 else
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
406 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
407 /* Protect against infinite loops. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
408 if (readlinks++ > MAX_READLINKS)
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
409 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
410 errno = ELOOP;
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
411 goto done;
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
412 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
413
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
414 /* Note: readlink doesn't add the null byte. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
415 link_path[n] = '\0';
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
416
1760
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
417 abslen = abs_start (link_path);
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
418 if (abslen > 0)
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
419 {
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
420 /* Start over for an absolute symlink. */
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
421 new_path = resolved_path;
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
422 qxestrcat (link_path, path);
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
423 path = link_path;
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
424 goto restart;
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
425 }
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
426
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
427 /* Otherwise back up over this component. */
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
428 for (--new_path; !IS_DIRECTORY_SEP (*new_path); --new_path)
0240ff815597 [xemacs-hg @ 2003-10-22 14:16:58 by james]
james
parents: 1204
diff changeset
429 assert (new_path > resolved_path);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
430
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
431 /* Safe sex check. */
2421
ab71ad6ff3dd [xemacs-hg @ 2004-12-06 03:50:53 by ben]
ben
parents: 2367
diff changeset
432 if (qxestrlen (path) + n >= PATH_MAX_INTERNAL)
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
433 {
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
434 errno = ENAMETOOLONG;
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
435 goto done;
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
436 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
437
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
438 /* Insert symlink contents into path. */
771
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
439 qxestrcat (link_path, path);
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
440 qxestrcpy (copy_path, link_path);
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
441 path = copy_path;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
442 }
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
443 #endif /* HAVE_READLINK || WIN32_ANY */
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
444 *new_path++ = DIRECTORY_SEP;
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
445 }
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
446
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
447 /* Delete trailing slash but don't whomp a lone slash. */
1116
3bcd77d0bf93 [xemacs-hg @ 2002-11-22 12:57:09 by ben]
ben
parents: 988
diff changeset
448 if (new_path != resolved_path + abs_start (resolved_path) &&
771
943eaba38521 [xemacs-hg @ 2002-03-13 08:51:24 by ben]
ben
parents: 593
diff changeset
449 IS_DIRECTORY_SEP (new_path[-1]))
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
450 new_path--;
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
451
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
452 /* Make sure it's null terminated. */
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
453 *new_path = '\0';
446
1ccc32a20af4 Import from CVS: tag r21-2-38
cvs
parents: 442
diff changeset
454
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
455 retval = resolved_path;
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
456 done:
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
457 PROFILE_RECORD_EXITING_SECTION (QSin_qxe_realpath);
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
458 return retval;
428
3ecd8885ac67 Import from CVS: tag r21-2-22
cvs
parents:
diff changeset
459 }
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
460
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
461 void
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
462 vars_of_realpath (void)
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
463 {
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
464 QSin_qxe_realpath =
4952
19a72041c5ed Mule-izing, various fixes related to char * arguments
Ben Wing <ben@xemacs.org>
parents: 4834
diff changeset
465 build_defer_string ("(in qxe_realpath)");
2526
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
466 staticpro (&QSin_qxe_realpath);
902d5bd9b75c [xemacs-hg @ 2005-01-28 02:36:11 by ben]
ben
parents: 2421
diff changeset
467 }