1*946379e7Schristos /* Execute a Java program.
2*946379e7Schristos Copyright (C) 2001-2003, 2006 Free Software Foundation, Inc.
3*946379e7Schristos Written by Bruno Haible <haible@clisp.cons.org>, 2001.
4*946379e7Schristos
5*946379e7Schristos This program is free software; you can redistribute it and/or modify
6*946379e7Schristos it under the terms of the GNU General Public License as published by
7*946379e7Schristos the Free Software Foundation; either version 2, or (at your option)
8*946379e7Schristos any later version.
9*946379e7Schristos
10*946379e7Schristos This program is distributed in the hope that it will be useful,
11*946379e7Schristos but WITHOUT ANY WARRANTY; without even the implied warranty of
12*946379e7Schristos MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13*946379e7Schristos GNU General Public License for more details.
14*946379e7Schristos
15*946379e7Schristos You should have received a copy of the GNU General Public License
16*946379e7Schristos along with this program; if not, write to the Free Software Foundation,
17*946379e7Schristos Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */
18*946379e7Schristos
19*946379e7Schristos #include <config.h>
20*946379e7Schristos #include <alloca.h>
21*946379e7Schristos
22*946379e7Schristos /* Specification. */
23*946379e7Schristos #include "javaexec.h"
24*946379e7Schristos
25*946379e7Schristos #include <stdio.h>
26*946379e7Schristos #include <stdlib.h>
27*946379e7Schristos #include <string.h>
28*946379e7Schristos
29*946379e7Schristos #include "execute.h"
30*946379e7Schristos #include "classpath.h"
31*946379e7Schristos #include "xsetenv.h"
32*946379e7Schristos #include "sh-quote.h"
33*946379e7Schristos #include "pathname.h"
34*946379e7Schristos #include "xalloc.h"
35*946379e7Schristos #include "xallocsa.h"
36*946379e7Schristos #include "error.h"
37*946379e7Schristos #include "gettext.h"
38*946379e7Schristos
39*946379e7Schristos #define _(str) gettext (str)
40*946379e7Schristos
41*946379e7Schristos
42*946379e7Schristos /* Survey of Java virtual machines.
43*946379e7Schristos
44*946379e7Schristos A = does it work without CLASSPATH being set
45*946379e7Schristos B = does it work with CLASSPATH being set to empty
46*946379e7Schristos C = option to set CLASSPATH, other than setting it in the environment
47*946379e7Schristos T = test for presence
48*946379e7Schristos
49*946379e7Schristos Program from A B C T
50*946379e7Schristos
51*946379e7Schristos $JAVA unknown N Y n/a true
52*946379e7Schristos gij GCC 3.0 Y Y n/a gij --version >/dev/null
53*946379e7Schristos java JDK 1.1.8 Y Y -classpath P java -version 2>/dev/null
54*946379e7Schristos jre JDK 1.1.8 N Y -classpath P jre 2>/dev/null; test $? = 1
55*946379e7Schristos java JDK 1.3.0 Y Y -classpath P java -version 2>/dev/null
56*946379e7Schristos jview MS IE Y Y -cp P jview -? >nul; %errorlevel% = 1
57*946379e7Schristos
58*946379e7Schristos The CLASSPATH is a colon separated list of pathnames. (On Windows: a
59*946379e7Schristos semicolon separated list of pathnames.)
60*946379e7Schristos
61*946379e7Schristos We try the Java virtual machines in the following order:
62*946379e7Schristos 1. getenv ("JAVA"), because the user must be able to override our
63*946379e7Schristos preferences,
64*946379e7Schristos 2. "gij", because it is a completely free JVM,
65*946379e7Schristos 3. "java", because it is a standard JVM,
66*946379e7Schristos 4. "jre", comes last because it requires a CLASSPATH environment variable,
67*946379e7Schristos 5. "jview", on Windows only, because it is frequently installed.
68*946379e7Schristos
69*946379e7Schristos We unset the JAVA_HOME environment variable, because a wrong setting of
70*946379e7Schristos this variable can confuse the JDK's javac.
71*946379e7Schristos */
72*946379e7Schristos
73*946379e7Schristos bool
execute_java_class(const char * class_name,const char * const * classpaths,unsigned int classpaths_count,bool use_minimal_classpath,const char * exe_dir,const char * const * args,bool verbose,bool quiet,execute_fn * executer,void * private_data)74*946379e7Schristos execute_java_class (const char *class_name,
75*946379e7Schristos const char * const *classpaths,
76*946379e7Schristos unsigned int classpaths_count,
77*946379e7Schristos bool use_minimal_classpath,
78*946379e7Schristos const char *exe_dir,
79*946379e7Schristos const char * const *args,
80*946379e7Schristos bool verbose, bool quiet,
81*946379e7Schristos execute_fn *executer, void *private_data)
82*946379e7Schristos {
83*946379e7Schristos bool err = false;
84*946379e7Schristos unsigned int nargs;
85*946379e7Schristos char *old_JAVA_HOME;
86*946379e7Schristos
87*946379e7Schristos /* Count args. */
88*946379e7Schristos {
89*946379e7Schristos const char * const *arg;
90*946379e7Schristos
91*946379e7Schristos for (nargs = 0, arg = args; *arg != NULL; nargs++, arg++)
92*946379e7Schristos ;
93*946379e7Schristos }
94*946379e7Schristos
95*946379e7Schristos /* First, try a class compiled to a native code executable. */
96*946379e7Schristos if (exe_dir != NULL)
97*946379e7Schristos {
98*946379e7Schristos char *exe_pathname = concatenated_pathname (exe_dir, class_name, EXEEXT);
99*946379e7Schristos char *old_classpath;
100*946379e7Schristos char **argv = (char **) xallocsa ((1 + nargs + 1) * sizeof (char *));
101*946379e7Schristos unsigned int i;
102*946379e7Schristos
103*946379e7Schristos /* Set CLASSPATH. */
104*946379e7Schristos old_classpath =
105*946379e7Schristos set_classpath (classpaths, classpaths_count, use_minimal_classpath,
106*946379e7Schristos verbose);
107*946379e7Schristos
108*946379e7Schristos argv[0] = exe_pathname;
109*946379e7Schristos for (i = 0; i <= nargs; i++)
110*946379e7Schristos argv[1 + i] = (char *) args[i];
111*946379e7Schristos
112*946379e7Schristos if (verbose)
113*946379e7Schristos {
114*946379e7Schristos char *command = shell_quote_argv (argv);
115*946379e7Schristos printf ("%s\n", command);
116*946379e7Schristos free (command);
117*946379e7Schristos }
118*946379e7Schristos
119*946379e7Schristos err = executer (class_name, exe_pathname, argv, private_data);
120*946379e7Schristos
121*946379e7Schristos /* Reset CLASSPATH. */
122*946379e7Schristos reset_classpath (old_classpath);
123*946379e7Schristos
124*946379e7Schristos freesa (argv);
125*946379e7Schristos
126*946379e7Schristos goto done1;
127*946379e7Schristos }
128*946379e7Schristos
129*946379e7Schristos {
130*946379e7Schristos const char *java = getenv ("JAVA");
131*946379e7Schristos if (java != NULL && java[0] != '\0')
132*946379e7Schristos {
133*946379e7Schristos /* Because $JAVA may consist of a command and options, we use the
134*946379e7Schristos shell. Because $JAVA has been set by the user, we leave all
135*946379e7Schristos all environment variables in place, including JAVA_HOME, and
136*946379e7Schristos we don't erase the user's CLASSPATH. */
137*946379e7Schristos char *old_classpath;
138*946379e7Schristos unsigned int command_length;
139*946379e7Schristos char *command;
140*946379e7Schristos char *argv[4];
141*946379e7Schristos const char * const *arg;
142*946379e7Schristos char *p;
143*946379e7Schristos
144*946379e7Schristos /* Set CLASSPATH. */
145*946379e7Schristos old_classpath =
146*946379e7Schristos set_classpath (classpaths, classpaths_count, false,
147*946379e7Schristos verbose);
148*946379e7Schristos
149*946379e7Schristos command_length = strlen (java);
150*946379e7Schristos command_length += 1 + shell_quote_length (class_name);
151*946379e7Schristos for (arg = args; *arg != NULL; arg++)
152*946379e7Schristos command_length += 1 + shell_quote_length (*arg);
153*946379e7Schristos command_length += 1;
154*946379e7Schristos
155*946379e7Schristos command = (char *) xallocsa (command_length);
156*946379e7Schristos p = command;
157*946379e7Schristos /* Don't shell_quote $JAVA, because it may consist of a command
158*946379e7Schristos and options. */
159*946379e7Schristos memcpy (p, java, strlen (java));
160*946379e7Schristos p += strlen (java);
161*946379e7Schristos *p++ = ' ';
162*946379e7Schristos p = shell_quote_copy (p, class_name);
163*946379e7Schristos for (arg = args; *arg != NULL; arg++)
164*946379e7Schristos {
165*946379e7Schristos *p++ = ' ';
166*946379e7Schristos p = shell_quote_copy (p, *arg);
167*946379e7Schristos }
168*946379e7Schristos *p++ = '\0';
169*946379e7Schristos /* Ensure command_length was correctly calculated. */
170*946379e7Schristos if (p - command > command_length)
171*946379e7Schristos abort ();
172*946379e7Schristos
173*946379e7Schristos if (verbose)
174*946379e7Schristos printf ("%s\n", command);
175*946379e7Schristos
176*946379e7Schristos argv[0] = "/bin/sh";
177*946379e7Schristos argv[1] = "-c";
178*946379e7Schristos argv[2] = command;
179*946379e7Schristos argv[3] = NULL;
180*946379e7Schristos err = executer (java, "/bin/sh", argv, private_data);
181*946379e7Schristos
182*946379e7Schristos freesa (command);
183*946379e7Schristos
184*946379e7Schristos /* Reset CLASSPATH. */
185*946379e7Schristos reset_classpath (old_classpath);
186*946379e7Schristos
187*946379e7Schristos goto done1;
188*946379e7Schristos }
189*946379e7Schristos }
190*946379e7Schristos
191*946379e7Schristos /* Unset the JAVA_HOME environment variable. */
192*946379e7Schristos old_JAVA_HOME = getenv ("JAVA_HOME");
193*946379e7Schristos if (old_JAVA_HOME != NULL)
194*946379e7Schristos {
195*946379e7Schristos old_JAVA_HOME = xstrdup (old_JAVA_HOME);
196*946379e7Schristos unsetenv ("JAVA_HOME");
197*946379e7Schristos }
198*946379e7Schristos
199*946379e7Schristos {
200*946379e7Schristos static bool gij_tested;
201*946379e7Schristos static bool gij_present;
202*946379e7Schristos
203*946379e7Schristos if (!gij_tested)
204*946379e7Schristos {
205*946379e7Schristos /* Test for presence of gij: "gij --version > /dev/null" */
206*946379e7Schristos char *argv[3];
207*946379e7Schristos int exitstatus;
208*946379e7Schristos
209*946379e7Schristos argv[0] = "gij";
210*946379e7Schristos argv[1] = "--version";
211*946379e7Schristos argv[2] = NULL;
212*946379e7Schristos exitstatus = execute ("gij", "gij", argv, false, false, true, true,
213*946379e7Schristos true, false);
214*946379e7Schristos gij_present = (exitstatus == 0);
215*946379e7Schristos gij_tested = true;
216*946379e7Schristos }
217*946379e7Schristos
218*946379e7Schristos if (gij_present)
219*946379e7Schristos {
220*946379e7Schristos char *old_classpath;
221*946379e7Schristos char **argv = (char **) xallocsa ((2 + nargs + 1) * sizeof (char *));
222*946379e7Schristos unsigned int i;
223*946379e7Schristos
224*946379e7Schristos /* Set CLASSPATH. */
225*946379e7Schristos old_classpath =
226*946379e7Schristos set_classpath (classpaths, classpaths_count, use_minimal_classpath,
227*946379e7Schristos verbose);
228*946379e7Schristos
229*946379e7Schristos argv[0] = "gij";
230*946379e7Schristos argv[1] = (char *) class_name;
231*946379e7Schristos for (i = 0; i <= nargs; i++)
232*946379e7Schristos argv[2 + i] = (char *) args[i];
233*946379e7Schristos
234*946379e7Schristos if (verbose)
235*946379e7Schristos {
236*946379e7Schristos char *command = shell_quote_argv (argv);
237*946379e7Schristos printf ("%s\n", command);
238*946379e7Schristos free (command);
239*946379e7Schristos }
240*946379e7Schristos
241*946379e7Schristos err = executer ("gij", "gij", argv, private_data);
242*946379e7Schristos
243*946379e7Schristos /* Reset CLASSPATH. */
244*946379e7Schristos reset_classpath (old_classpath);
245*946379e7Schristos
246*946379e7Schristos freesa (argv);
247*946379e7Schristos
248*946379e7Schristos goto done2;
249*946379e7Schristos }
250*946379e7Schristos }
251*946379e7Schristos
252*946379e7Schristos {
253*946379e7Schristos static bool java_tested;
254*946379e7Schristos static bool java_present;
255*946379e7Schristos
256*946379e7Schristos if (!java_tested)
257*946379e7Schristos {
258*946379e7Schristos /* Test for presence of java: "java -version 2> /dev/null" */
259*946379e7Schristos char *argv[3];
260*946379e7Schristos int exitstatus;
261*946379e7Schristos
262*946379e7Schristos argv[0] = "java";
263*946379e7Schristos argv[1] = "-version";
264*946379e7Schristos argv[2] = NULL;
265*946379e7Schristos exitstatus = execute ("java", "java", argv, false, false, true, true,
266*946379e7Schristos true, false);
267*946379e7Schristos java_present = (exitstatus == 0);
268*946379e7Schristos java_tested = true;
269*946379e7Schristos }
270*946379e7Schristos
271*946379e7Schristos if (java_present)
272*946379e7Schristos {
273*946379e7Schristos char *old_classpath;
274*946379e7Schristos char **argv = (char **) xallocsa ((2 + nargs + 1) * sizeof (char *));
275*946379e7Schristos unsigned int i;
276*946379e7Schristos
277*946379e7Schristos /* Set CLASSPATH. We don't use the "-classpath ..." option because
278*946379e7Schristos in JDK 1.1.x its argument should also contain the JDK's classes.zip,
279*946379e7Schristos but we don't know its location. (In JDK 1.3.0 it would work.) */
280*946379e7Schristos old_classpath =
281*946379e7Schristos set_classpath (classpaths, classpaths_count, use_minimal_classpath,
282*946379e7Schristos verbose);
283*946379e7Schristos
284*946379e7Schristos argv[0] = "java";
285*946379e7Schristos argv[1] = (char *) class_name;
286*946379e7Schristos for (i = 0; i <= nargs; i++)
287*946379e7Schristos argv[2 + i] = (char *) args[i];
288*946379e7Schristos
289*946379e7Schristos if (verbose)
290*946379e7Schristos {
291*946379e7Schristos char *command = shell_quote_argv (argv);
292*946379e7Schristos printf ("%s\n", command);
293*946379e7Schristos free (command);
294*946379e7Schristos }
295*946379e7Schristos
296*946379e7Schristos err = executer ("java", "java", argv, private_data);
297*946379e7Schristos
298*946379e7Schristos /* Reset CLASSPATH. */
299*946379e7Schristos reset_classpath (old_classpath);
300*946379e7Schristos
301*946379e7Schristos freesa (argv);
302*946379e7Schristos
303*946379e7Schristos goto done2;
304*946379e7Schristos }
305*946379e7Schristos }
306*946379e7Schristos
307*946379e7Schristos {
308*946379e7Schristos static bool jre_tested;
309*946379e7Schristos static bool jre_present;
310*946379e7Schristos
311*946379e7Schristos if (!jre_tested)
312*946379e7Schristos {
313*946379e7Schristos /* Test for presence of jre: "jre 2> /dev/null ; test $? = 1" */
314*946379e7Schristos char *argv[2];
315*946379e7Schristos int exitstatus;
316*946379e7Schristos
317*946379e7Schristos argv[0] = "jre";
318*946379e7Schristos argv[1] = NULL;
319*946379e7Schristos exitstatus = execute ("jre", "jre", argv, false, false, true, true,
320*946379e7Schristos true, false);
321*946379e7Schristos jre_present = (exitstatus == 0 || exitstatus == 1);
322*946379e7Schristos jre_tested = true;
323*946379e7Schristos }
324*946379e7Schristos
325*946379e7Schristos if (jre_present)
326*946379e7Schristos {
327*946379e7Schristos char *old_classpath;
328*946379e7Schristos char **argv = (char **) xallocsa ((2 + nargs + 1) * sizeof (char *));
329*946379e7Schristos unsigned int i;
330*946379e7Schristos
331*946379e7Schristos /* Set CLASSPATH. We don't use the "-classpath ..." option because
332*946379e7Schristos in JDK 1.1.x its argument should also contain the JDK's classes.zip,
333*946379e7Schristos but we don't know its location. */
334*946379e7Schristos old_classpath =
335*946379e7Schristos set_classpath (classpaths, classpaths_count, use_minimal_classpath,
336*946379e7Schristos verbose);
337*946379e7Schristos
338*946379e7Schristos argv[0] = "jre";
339*946379e7Schristos argv[1] = (char *) class_name;
340*946379e7Schristos for (i = 0; i <= nargs; i++)
341*946379e7Schristos argv[2 + i] = (char *) args[i];
342*946379e7Schristos
343*946379e7Schristos if (verbose)
344*946379e7Schristos {
345*946379e7Schristos char *command = shell_quote_argv (argv);
346*946379e7Schristos printf ("%s\n", command);
347*946379e7Schristos free (command);
348*946379e7Schristos }
349*946379e7Schristos
350*946379e7Schristos err = executer ("jre", "jre", argv, private_data);
351*946379e7Schristos
352*946379e7Schristos /* Reset CLASSPATH. */
353*946379e7Schristos reset_classpath (old_classpath);
354*946379e7Schristos
355*946379e7Schristos freesa (argv);
356*946379e7Schristos
357*946379e7Schristos goto done2;
358*946379e7Schristos }
359*946379e7Schristos }
360*946379e7Schristos
361*946379e7Schristos #if defined _WIN32 || defined __WIN32__ || defined __CYGWIN__
362*946379e7Schristos /* Win32, Cygwin */
363*946379e7Schristos {
364*946379e7Schristos static bool jview_tested;
365*946379e7Schristos static bool jview_present;
366*946379e7Schristos
367*946379e7Schristos if (!jview_tested)
368*946379e7Schristos {
369*946379e7Schristos /* Test for presence of jview: "jview -? >nul ; test $? = 1" */
370*946379e7Schristos char *argv[3];
371*946379e7Schristos int exitstatus;
372*946379e7Schristos
373*946379e7Schristos argv[0] = "jview";
374*946379e7Schristos argv[1] = "-?";
375*946379e7Schristos argv[2] = NULL;
376*946379e7Schristos exitstatus = execute ("jview", "jview", argv, false, false, true, true,
377*946379e7Schristos true, false);
378*946379e7Schristos jview_present = (exitstatus == 0 || exitstatus == 1);
379*946379e7Schristos jview_tested = true;
380*946379e7Schristos }
381*946379e7Schristos
382*946379e7Schristos if (jview_present)
383*946379e7Schristos {
384*946379e7Schristos char *old_classpath;
385*946379e7Schristos char **argv = (char **) xallocsa ((2 + nargs + 1) * sizeof (char *));
386*946379e7Schristos unsigned int i;
387*946379e7Schristos
388*946379e7Schristos /* Set CLASSPATH. */
389*946379e7Schristos old_classpath =
390*946379e7Schristos set_classpath (classpaths, classpaths_count, use_minimal_classpath,
391*946379e7Schristos verbose);
392*946379e7Schristos
393*946379e7Schristos argv[0] = "jview";
394*946379e7Schristos argv[1] = (char *) class_name;
395*946379e7Schristos for (i = 0; i <= nargs; i++)
396*946379e7Schristos argv[2 + i] = (char *) args[i];
397*946379e7Schristos
398*946379e7Schristos if (verbose)
399*946379e7Schristos {
400*946379e7Schristos char *command = shell_quote_argv (argv);
401*946379e7Schristos printf ("%s\n", command);
402*946379e7Schristos free (command);
403*946379e7Schristos }
404*946379e7Schristos
405*946379e7Schristos err = executer ("jview", "jview", argv, private_data);
406*946379e7Schristos
407*946379e7Schristos /* Reset CLASSPATH. */
408*946379e7Schristos reset_classpath (old_classpath);
409*946379e7Schristos
410*946379e7Schristos freesa (argv);
411*946379e7Schristos
412*946379e7Schristos goto done2;
413*946379e7Schristos }
414*946379e7Schristos }
415*946379e7Schristos #endif
416*946379e7Schristos
417*946379e7Schristos if (!quiet)
418*946379e7Schristos error (0, 0, _("Java virtual machine not found, try installing gij or set $JAVA"));
419*946379e7Schristos err = true;
420*946379e7Schristos
421*946379e7Schristos done2:
422*946379e7Schristos if (old_JAVA_HOME != NULL)
423*946379e7Schristos {
424*946379e7Schristos xsetenv ("JAVA_HOME", old_JAVA_HOME, 1);
425*946379e7Schristos free (old_JAVA_HOME);
426*946379e7Schristos }
427*946379e7Schristos
428*946379e7Schristos done1:
429*946379e7Schristos return err;
430*946379e7Schristos }
431