1 /* Handle CLASSPATH, -classpath, and path searching.
3 Copyright (C) 1998 Free Software Foundation, Inc.
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2, or (at your option)
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with GNU CC; see the file COPYING. If not, write to
17 the Free Software Foundation, 59 Temple Place - Suite 330,
18 Boston, MA 02111-1307, USA.
20 Java and all Java-based marks are trademarks or registered trademarks
21 of Sun Microsystems, Inc. in the United States and other countries.
22 The Free Software Foundation is independent of Sun Microsystems, Inc. */
24 /* Written by Tom Tromey <tromey@cygnus.com>, October 1998. */
31 /* Some boilerplate that really belongs in a header. */
33 #ifndef GET_ENV_PATH_LIST
34 #define GET_ENV_PATH_LIST(VAR,NAME) do { (VAR) = getenv (NAME); } while (0)
37 /* By default, colon separates directories in a path. */
38 #ifndef PATH_SEPARATOR
39 #define PATH_SEPARATOR ':'
43 #define DIR_SEPARATOR '/'
48 /* Possible flag values. */
52 /* We keep linked lists of directory names. A ``directory'' can be
53 either an ordinary directory or a .zip file. */
61 /* We support several different ways to set the class path.
63 built-in system directory (only libjava.zip)
64 CLASSPATH environment variable
65 -CLASSPATH overrides CLASSPATH
66 -classpath option - overrides CLASSPATH, -CLASSPATH, and built-in
67 -I prepends path to list
69 We implement this by keeping several path lists, and then simply
70 ignoring the ones which are not relevant. */
72 /* This holds all the -I directories. */
73 static struct entry
*include_dirs
;
75 /* This holds the CLASSPATH environment variable. */
76 static struct entry
*classpath_env
;
78 /* This holds the -CLASSPATH command-line option. */
79 static struct entry
*classpath_u
;
81 /* This holds the -classpath command-line option. */
82 static struct entry
*classpath_l
;
84 /* This holds the default directories. Some of these will have the
86 static struct entry
*sys_dirs
;
88 /* This is the sealed list. It is just a combination of other lists. */
89 static struct entry
*sealed
;
91 /* We keep track of the longest path we've seen. */
92 static int longest_path
= 0;
102 for (e
= *entp
; e
; e
= n
)
112 append_entry (entp
, ent
)
116 /* It doesn't matter if this is slow, since it is run only at
117 startup, and then infrequently. */
120 /* Find end of list. */
121 for (e
= *entp
; e
&& e
->next
; e
= e
->next
)
131 add_entry (entp
, filename
, is_system
)
139 n
= (struct entry
*) ALLOC (sizeof (struct entry
));
140 n
->flags
= is_system
? FLAG_SYSTEM
: 0;
143 len
= strlen (filename
);
144 if (len
> 4 && ! strcmp (filename
- 4, ".zip"))
146 n
->flags
|= FLAG_ZIP
;
147 /* If the user uses -classpath then he'll have to include
148 libjava.zip in the value. We check for this in a simplistic
149 way. Symlinks will fool this test. This is only used for
150 -MM and -MMD, so it probably isn't terribly important. */
151 if (! strcmp (filename
, LIBJAVA_ZIP_FILE
))
152 n
->flags
|= FLAG_SYSTEM
;
155 if (filename
[len
- 1] != '/' && filename
[len
- 1] != DIR_SEPARATOR
)
157 char *f2
= (char *) alloca (len
+ 1);
158 strcpy (f2
, filename
);
159 f2
[len
] = DIR_SEPARATOR
;
161 n
->name
= strdup (f2
);
165 n
->name
= strdup (filename
);
167 if (len
> longest_path
)
170 append_entry (entp
, n
);
174 add_path (entp
, cp
, is_system
)
183 char *buf
= (char *) alloca (strlen (cp
) + 3);
187 if (! *endp
|| *endp
== PATH_SEPARATOR
)
189 strncpy (buf
, startp
, endp
- startp
);
193 buf
[1] = DIR_SEPARATOR
;
196 else if (endp
[-1] != '/' && endp
[1] != DIR_SEPARATOR
)
198 buf
[endp
- startp
] = DIR_SEPARATOR
;
199 buf
[endp
- startp
+ 1] = '\0';
202 buf
[endp
- startp
] = '\0';
203 add_entry (entp
, buf
, is_system
);
215 /* Initialize the path module. */
221 add_entry (&sys_dirs
, ".", 0);
222 add_entry (&sys_dirs
, LIBJAVA_ZIP_FILE
, 1);
224 GET_ENV_PATH_LIST (cp
, "CLASSPATH");
225 add_path (&classpath_env
, cp
, 0);
228 /* Call this when -classpath is seen on the command line. */
230 jcf_path_classpath_arg (path
)
233 free_entry (&classpath_l
);
234 add_path (&classpath_l
, path
, 0);
237 /* Call this when -CLASSPATH is seen on the command line. */
239 jcf_path_CLASSPATH_arg (path
)
242 free_entry (&classpath_u
);
243 add_path (&classpath_u
, path
, 0);
246 /* Call this when -I is seen on the command line. */
248 jcf_path_include_arg (path
)
251 add_entry (&include_dirs
, path
, 0);
254 /* We `seal' the path by linking everything into one big list. Then
255 we provide a way to iterate through the sealed list. */
260 struct entry
*secondary
;
262 sealed
= include_dirs
;
267 secondary
= classpath_l
;
271 else if (classpath_u
)
273 secondary
= classpath_u
;
278 secondary
= classpath_env
;
279 classpath_env
= NULL
;
282 free_entry (&classpath_l
);
283 free_entry (&classpath_u
);
284 free_entry (&classpath_env
);
286 append_entry (&sealed
, secondary
);
290 append_entry (&sealed
, sys_dirs
);
294 free_entry (&sys_dirs
);
300 return (void *) sealed
;
307 struct entry
*ent
= (struct entry
*) x
;
308 return (void *) ent
->next
;
311 /* We guarantee that the return path will either be a zip file, or it
312 will end with a directory separator. */
317 struct entry
*ent
= (struct entry
*) x
;
322 jcf_path_is_zipfile (x
)
325 struct entry
*ent
= (struct entry
*) x
;
326 return (ent
->flags
& FLAG_ZIP
);
330 jcf_path_is_system (x
)
333 struct entry
*ent
= (struct entry
*) x
;
334 return (ent
->flags
& FLAG_SYSTEM
);