annotate src/realpath.c @ 5872:f9e59cd39a9a

Clean up #'read-quoted-char, support help-event-list there. lisp/changeLog addition: 2015-03-14 Aidan Kehoe <kehoea@parhasard.net> * simple.el (quoted-insert): Update the docstring here, syncing GNU's, especially mentioning read-quoted-char-radix. * cmdloop.el: * cmdloop.el (read-quoted-char-radix): Move this up here, outside the functions. * cmdloop.el (read-function-key-map): New label, reading and replacing characters from function-key-map if appropriate. * cmdloop.el (read-quoted-char): Multiple changes: -- Take advantage of help-event-list, but be careful not to have any keystrokes with character equivalents in it, so the user can type C-q C-h and have the expected result. -- Use function-key-map, as does #'read-char and #'read-exclusive-char, helpful for character composition under X11. -- Pop up the help window ourselves if, e.g. F1 arrives on a TTY via function-key-map, event-stream won't have done it. -- Error if no keystroke that can be converted into a character is specified, don't just insert ?\x00 as we used to and as does GNU -- Use #'digit-char-p instead of reimplementing it. -- Fix a bug of mine where I wasn't consistent about treating character codes as Unicode.
author Aidan Kehoe <kehoea@parhasard.net>
date Tue, 17 Mar 2015 12:22:50 +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 }