/* Handle CLASSPATH, -classpath, and path searching. Copyright (C) 1998, 1999, 2000, 2001 Free Software Foundation, Inc. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with GNU CC; see the file COPYING. If not, write to the Free Software Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. Java and all Java-based marks are trademarks or registered trademarks of Sun Microsystems, Inc. in the United States and other countries. The Free Software Foundation is independent of Sun Microsystems, Inc. */ /* Written by Tom Tromey , October 1998. */ #include "config.h" #include "system.h" #include "jcf.h" /* Some boilerplate that really belongs in a header. */ #ifndef GET_ENV_PATH_LIST #define GET_ENV_PATH_LIST(VAR,NAME) do { (VAR) = getenv (NAME); } while (0) #endif /* By default, colon separates directories in a path. */ #ifndef PATH_SEPARATOR #define PATH_SEPARATOR ':' #endif #ifndef DIR_SEPARATOR #define DIR_SEPARATOR '/' #endif #ifndef DIR_UP #define DIR_UP ".." #endif /* Possible flag values. */ #define FLAG_SYSTEM 1 #define FLAG_ZIP 2 /* We keep linked lists of directory names. A ``directory'' can be either an ordinary directory or a .zip file. */ struct entry { char *name; int flags; struct entry *next; }; static void free_entry PARAMS ((struct entry **)); static void append_entry PARAMS ((struct entry **, struct entry *)); static void add_entry PARAMS ((struct entry **, const char *, int)); static void add_path PARAMS ((struct entry **, const char *, int)); /* We support several different ways to set the class path. built-in system directory (only libgcj.jar) CLASSPATH environment variable -classpath option overrides $CLASSPATH -CLASSPATH option overrides $CLASSPATH, -classpath, and built-in -I prepends path to list We implement this by keeping several path lists, and then simply ignoring the ones which are not relevant. */ /* This holds all the -I directories. */ static struct entry *include_dirs; /* This holds the CLASSPATH environment variable. */ static struct entry *classpath_env; /* This holds the -classpath command-line option. */ static struct entry *classpath_u; /* This holds the -CLASSPATH command-line option. */ static struct entry *classpath_l; /* This holds the default directories. Some of these will have the "system" flag set. */ static struct entry *sys_dirs; /* This is the sealed list. It is just a combination of other lists. */ static struct entry *sealed; /* We keep track of the longest path we've seen. */ static int longest_path = 0; static void free_entry (entp) struct entry **entp; { struct entry *e, *n; for (e = *entp; e; e = n) { n = e->next; free (e->name); free (e); } *entp = NULL; } static void append_entry (entp, ent) struct entry **entp; struct entry *ent; { /* It doesn't matter if this is slow, since it is run only at startup, and then infrequently. */ struct entry *e; /* Find end of list. */ for (e = *entp; e && e->next; e = e->next) ; if (e) e->next = ent; else *entp = ent; } static void add_entry (entp, filename, is_system) struct entry **entp; const char *filename; int is_system; { int len; struct entry *n; n = (struct entry *) ALLOC (sizeof (struct entry)); n->flags = is_system ? FLAG_SYSTEM : 0; n->next = NULL; len = strlen (filename); if (len > 4 && (strcmp (filename + len - 4, ".zip") == 0 || strcmp (filename + len - 4, ".jar") == 0)) { n->flags |= FLAG_ZIP; /* If the user uses -classpath then he'll have to include libgcj.jar in the value. We check for this in a simplistic way. Symlinks will fool this test. This is only used for -MM and -MMD, so it probably isn't terribly important. */ if (! strcmp (filename, LIBGCJ_ZIP_FILE)) n->flags |= FLAG_SYSTEM; } /* Note that we add a trailing separator to `.zip' names as well. This is a little hack that lets the searching code in jcf-io.c work more easily. Eww. */ if (filename[len - 1] != '/' && filename[len - 1] != DIR_SEPARATOR) { char *f2 = (char *) alloca (len + 2); strcpy (f2, filename); f2[len] = DIR_SEPARATOR; f2[len + 1] = '\0'; n->name = xstrdup (f2); ++len; } else n->name = xstrdup (filename); if (len > longest_path) longest_path = len; append_entry (entp, n); } static void add_path (entp, cp, is_system) struct entry **entp; const char *cp; int is_system; { const char *startp, *endp; if (cp) { char *buf = (char *) alloca (strlen (cp) + 3); startp = endp = cp; while (1) { if (! *endp || *endp == PATH_SEPARATOR) { if (endp == startp) { buf[0] = '.'; buf[1] = DIR_SEPARATOR; buf[2] = '\0'; } else { strncpy (buf, startp, endp - startp); buf[endp - startp] = '\0'; } add_entry (entp, buf, is_system); if (! *endp) break; ++endp; startp = endp; } else ++endp; } } } /* Initialize the path module. */ void jcf_path_init () { char *cp; char *try, sep[2]; struct stat stat_b; int found = 0, len; add_entry (&sys_dirs, ".", 0); sep[0] = DIR_SEPARATOR; sep[1] = '\0'; GET_ENV_PATH_LIST (cp, "GCC_EXEC_PREFIX"); if (cp) { try = alloca (strlen (cp) + 50); /* The exec prefix can be something like /usr/local/bin/../lib/gcc-lib/. We want to change this into a pointer to the share directory. We support two configurations: one where prefix and exec-prefix are the same, and one where exec-prefix is `prefix/SOMETHING'. */ strcpy (try, cp); strcat (try, DIR_UP); strcat (try, sep); strcat (try, DIR_UP); strcat (try, sep); len = strlen (try); strcpy (try + len, "share"); strcat (try, sep); strcat (try, "libgcj.jar"); if (! stat (try, &stat_b)) { add_entry (&sys_dirs, try, 1); found = 1; } else { strcpy (try + len, DIR_UP); strcat (try, sep); strcat (try, "share"); strcat (try, sep); strcat (try, "libgcj.jar"); if (! stat (try, &stat_b)) { add_entry (&sys_dirs, try, 1); found = 1; } } } if (! found) { /* Desperation: use the installed one. */ add_entry (&sys_dirs, LIBGCJ_ZIP_FILE, 1); } GET_ENV_PATH_LIST (cp, "CLASSPATH"); add_path (&classpath_env, cp, 0); } /* Call this when -CLASSPATH is seen on the command line. This is the override-all switch, even the built in classes are overridden. */ void jcf_path_CLASSPATH_arg (path) const char *path; { free_entry (&classpath_l); add_path (&classpath_l, path, 0); } /* Call this when -classpath is seen on the command line. This overrides only the $CLASSPATH environment variable. */ void jcf_path_classpath_arg (path) const char *path; { free_entry (&classpath_u); add_path (&classpath_u, path, 0); } /* Call this when -I is seen on the command line. */ void jcf_path_include_arg (path) const char *path; { add_entry (&include_dirs, path, 0); } /* We `seal' the path by linking everything into one big list. Then we provide a way to iterate through the sealed list. If PRINT is true then we print the final class path to stderr. */ void jcf_path_seal (print) int print; { int do_system = 1; struct entry *secondary; sealed = include_dirs; include_dirs = NULL; if (classpath_l) { secondary = classpath_l; classpath_l = NULL; do_system = 0; } else if (classpath_u) { secondary = classpath_u; classpath_u = NULL; } else { secondary = classpath_env; classpath_env = NULL; } free_entry (&classpath_l); free_entry (&classpath_u); free_entry (&classpath_env); append_entry (&sealed, secondary); if (do_system) { append_entry (&sealed, sys_dirs); sys_dirs = NULL; } else free_entry (&sys_dirs); if (print) { struct entry *ent; fprintf (stderr, "Class path starts here:\n"); for (ent = sealed; ent; ent = ent->next) { fprintf (stderr, " %s", ent->name); if ((ent->flags & FLAG_SYSTEM)) fprintf (stderr, " (system)"); if ((ent->flags & FLAG_ZIP)) fprintf (stderr, " (zip)"); fprintf (stderr, "\n"); } } } void * jcf_path_start () { return (void *) sealed; } void * jcf_path_next (x) void *x; { struct entry *ent = (struct entry *) x; return (void *) ent->next; } /* We guarantee that the return path will either be a zip file, or it will end with a directory separator. */ char * jcf_path_name (x) void *x; { struct entry *ent = (struct entry *) x; return ent->name; } int jcf_path_is_zipfile (x) void *x; { struct entry *ent = (struct entry *) x; return (ent->flags & FLAG_ZIP); } int jcf_path_is_system (x) void *x; { struct entry *ent = (struct entry *) x; return (ent->flags & FLAG_SYSTEM); } int jcf_path_max_len () { return longest_path; }