Contents of release-21-2 at 1999-07-05-18.
[chise/xemacs-chise.git.1] / src / emacs.c
1 /* XEmacs -- Fully extensible Emacs, running on Unix and other platforms.
2    Copyright (C) 1985, 1986, 1987, 1992, 1993, 1994
3    Free Software Foundation, Inc.
4    Copyright (C) 1995 Sun Microsystems, Inc.
5
6 This file is part of XEmacs.
7
8 XEmacs is free software; you can redistribute it and/or modify it
9 under the terms of the GNU General Public License as published by the
10 Free Software Foundation; either version 2, or (at your option) any
11 later version.
12
13 XEmacs is distributed in the hope that it will be useful, but WITHOUT
14 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
15 FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
16 for more details.
17
18 You should have received a copy of the GNU General Public License
19 along with XEmacs; see the file COPYING.  If not, write to
20 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
21 Boston, MA 02111-1307, USA.  */
22
23 /* Synched up with: Mule 2.0, FSF 19.28. */
24
25 /* Note: It is necessary to specify <config.h> and not "config.h" in
26    order for the --srcdir type of compilation to work properly.
27    Otherwise the config.h from the srcdir, rather than the one from
28    the build dir, will be used. */
29
30 #include <config.h>
31 #include "lisp.h"
32
33 #include "backtrace.h" /* run-emacs-from-temacs needs this */
34 #include "buffer.h"
35 #include "commands.h"
36 #include "console.h"
37 #include "process.h"
38 #include "redisplay.h"
39 #include "sysdep.h"
40
41 #include "syssignal.h" /* Always include before systty.h */
42 #include "systty.h"
43 #include "sysfile.h"
44 #include "systime.h"
45
46 #ifdef QUANTIFY
47 #include <quantify.h>
48 #endif
49
50 #ifdef HAVE_SHLIB
51 #include "sysdll.h"
52 #endif
53
54 #if defined (HAVE_LOCALE_H) && \
55    (defined (I18N2) || defined (I18N3) || defined (I18N4))
56 #include <locale.h>
57 #endif
58
59 #ifdef TOOLTALK
60 #include TT_C_H_PATH
61 #endif
62
63 #ifdef APOLLO
64 #ifndef APOLLO_SR10
65 #include <default_acl.h>
66 #endif
67 #endif
68
69 #if defined (WINDOWSNT)
70 #include <windows.h>
71 #endif
72
73 /* For PATH_EXEC */
74 #include <paths.h>
75
76 #ifdef HEAP_IN_DATA
77 void report_sheap_usage (int die_if_pure_storage_exceeded);
78 #endif
79
80 #if !defined (SYSTEM_MALLOC) && !defined (DOUG_LEA_MALLOC)
81 extern void *(*__malloc_hook)(size_t);
82 extern void *(*__realloc_hook)(void *, size_t);
83 extern void (*__free_hook)(void *);
84 #endif  /* not SYSTEM_MALLOC && not DOUG_LEA_MALLOC */
85
86 /* Command line args from shell, as list of strings */
87 Lisp_Object Vcommand_line_args;
88
89 /* Set nonzero after XEmacs has started up the first time.
90   Prevents reinitialization of the Lisp world and keymaps
91   on subsequent starts.  */
92 int initialized;
93
94 #ifdef DOUG_LEA_MALLOC
95 # include <malloc.h>
96 /* Preserves a pointer to the memory allocated that copies that
97    static data inside glibc's malloc.  */
98 static void *malloc_state_ptr;
99 #endif /* DOUG_LEA_MALLOC */
100
101 # ifdef REL_ALLOC
102 void r_alloc_reinit (void);
103 # endif
104
105 /* Variable whose value is symbol giving operating system type. */
106 Lisp_Object Vsystem_type;
107
108 /* Variable whose value is string giving configuration built for.  */
109 Lisp_Object Vsystem_configuration;
110
111 /* Variable whose value is string containing the configuration options
112    XEmacs was built with.  */
113 Lisp_Object Vsystem_configuration_options;
114
115 /* Version numbers and strings */
116 Lisp_Object Vemacs_major_version;
117 Lisp_Object Vemacs_minor_version;
118 Lisp_Object Vemacs_patch_level;
119 Lisp_Object Vemacs_beta_version;
120 Lisp_Object Vxemacs_codename;
121 #ifdef INFODOCK
122 Lisp_Object Vinfodock_major_version;
123 Lisp_Object Vinfodock_minor_version;
124 Lisp_Object Vinfodock_build_version;
125 #endif
126
127 /* The path under which XEmacs was invoked. */
128 Lisp_Object Vinvocation_path;
129
130 /* The name under which XEmacs was invoked, with any leading directory
131    names discarded.  */
132 Lisp_Object Vinvocation_name;
133
134 /* The directory name from which XEmacs was invoked.  */
135 Lisp_Object Vinvocation_directory;
136
137 #if 0 /* FSFmacs */
138 /* The directory name in which to find subdirs such as lisp and etc.
139    nil means get them only from PATH_LOADSEARCH.  */
140 Lisp_Object Vinstallation_directory;
141 #endif
142
143 Lisp_Object Vemacs_program_name, Vemacs_program_version;
144 Lisp_Object Vexec_path;
145 Lisp_Object Vexec_directory, Vconfigure_exec_directory;
146 Lisp_Object Vlisp_directory, Vconfigure_lisp_directory;
147 Lisp_Object Vmodule_directory, Vconfigure_module_directory;
148 Lisp_Object Vsite_module_directory, Vconfigure_site_module_directory;
149 Lisp_Object Vconfigure_package_path;
150 Lisp_Object Vdata_directory, Vconfigure_data_directory;
151 Lisp_Object Vdoc_directory, Vconfigure_doc_directory;
152 Lisp_Object Vconfigure_lock_directory;
153 Lisp_Object Vdata_directory_list;
154 Lisp_Object Vinfo_directory, Vconfigure_info_directory;
155 Lisp_Object Vsite_directory, Vconfigure_site_directory;
156 Lisp_Object Vconfigure_info_path;
157 Lisp_Object Vinternal_error_checking;
158 Lisp_Object Vpath_separator;
159
160 /* The default base directory XEmacs is installed under. */
161 Lisp_Object Vconfigure_exec_prefix_directory, Vconfigure_prefix_directory;
162
163 /* If nonzero, set XEmacs to run at this priority.  This is also used
164    in child_setup and sys_suspend to make sure subshells run at normal
165    priority. */
166 int emacs_priority;
167
168 /* If non-zero a filter or a sentinel is running.  Tested to save the match
169    data on the first attempt to change it inside asynchronous code. */
170 int running_asynch_code;
171
172 /* If non-zero, a window-system was specified on the command line. */
173 int display_arg;
174
175 /* Type of display specified.  We cannot use a Lisp symbol here because
176    Lisp symbols may not initialized at the time that we set this
177    variable. */
178 CONST char *display_use;
179
180 /* If non-zero, then the early error handler will only print the error
181    message and exit. */
182 int suppress_early_error_handler_backtrace;
183
184 /* An address near the bottom of the stack.
185    Tells GC how to save a copy of the stack.  */
186 char *stack_bottom;
187
188 #ifdef USG_SHARED_LIBRARIES
189 /* If nonzero, this is the place to put the end of the writable segment
190    at startup.  */
191
192 uintptr_t bss_end = 0;
193 #endif
194
195 /* Number of bytes of writable memory we can expect to be able to get */
196 unsigned int lim_data;
197
198 /* Nonzero means running XEmacs without interactive terminal.  */
199
200 int noninteractive;
201
202 /* Value of Lisp variable `noninteractive'.
203    Normally same as C variable `noninteractive'
204    but nothing terrible happens if user sets this one.  */
205
206 int noninteractive1;
207
208 /* Nonzero means don't perform site-lisp searches at startup */
209 int inhibit_site_lisp;
210
211 /* Nonzero means don't perform site-modules searches at startup */
212 int inhibit_site_modules;
213
214 /* Nonzero means don't respect early packages at startup */
215 int inhibit_early_packages;
216
217 /* Nonzero means don't load package autoloads at startup */
218 int inhibit_autoloads;
219
220 /* Nonzero means print debug information about path searching */
221 int debug_paths;
222
223 /* Save argv and argc.  */
224 char **initial_argv;
225 int initial_argc;
226
227 static void sort_args (int argc, char **argv);
228
229 Lisp_Object Qkill_emacs_hook;
230 Lisp_Object Qsave_buffers_kill_emacs;
231
232 extern Lisp_Object Vlisp_EXEC_SUFFIXES;
233
234 \f
235 /* Signal code for the fatal signal that was received */
236 static int fatal_error_code;
237
238 /* Nonzero if handling a fatal error already */
239 static int fatal_error_in_progress;
240
241 static void shut_down_emacs (int sig, Lisp_Object stuff);
242
243 /* Handle bus errors, illegal instruction, etc. */
244 SIGTYPE
245 fatal_error_signal (int sig)
246 {
247   fatal_error_code = sig;
248   signal (sig, SIG_DFL);
249   /* Unblock the signal so that if the same signal gets sent in the
250      code below, we avoid a deadlock. */
251   EMACS_UNBLOCK_SIGNAL (fatal_error_code);
252
253   /* If fatal error occurs in code below, avoid infinite recursion.  */
254   if (! fatal_error_in_progress)
255     {
256       fatal_error_in_progress = dont_check_for_quit = 1;
257       shut_down_emacs (sig, Qnil);
258       stderr_out ("\nLisp backtrace follows:\n\n");
259       Fbacktrace (Qexternal_debugging_output, Qt);
260 # if 0  /* This is evil, rarely useful, and causes grief in some cases. */
261       /* Check for Sun-style stack printing via /proc */
262       {
263         CONST char *pstack = "/usr/proc/bin/pstack";
264         if (access (pstack, X_OK) == 0)
265           {
266             char buf[100];
267             stderr_out ("\nC backtrace follows:\n"
268                        "(A real debugger may provide better information)\n\n");
269             sprintf (buf, "%s %d >&2", pstack, (int)getpid());
270             system (buf);
271           }
272       }
273 # endif
274     }
275   /* Signal the same code; this time it will really be fatal. */
276   kill (getpid (), fatal_error_code);
277   SIGRETURN;
278 }
279 \f
280
281 DOESNT_RETURN
282 fatal (CONST char *fmt, ...)
283 {
284   va_list args;
285   va_start (args, fmt);
286
287   fprintf (stderr, "\nXEmacs: ");
288   vfprintf (stderr, GETTEXT (fmt), args);
289   fprintf (stderr, "\n");
290
291   va_end (args);
292   fflush (stderr);
293   exit (1);
294 }
295
296 /* #### The following two functions should be replaced with
297    calls to emacs_doprnt_*() functions, with STREAM set to send out
298    to stdout or stderr.  This is the only way to ensure that
299    I18N3 works properly (many implementations of the *printf()
300    functions, including the ones included in glibc, do not implement
301    the %###$ argument-positioning syntax). */
302
303 /* exactly equivalent to fprintf (stderr, fmt, ...) except that it calls
304    GETTEXT on the format string. */
305
306 int
307 stderr_out (CONST char *fmt, ...)
308 {
309   int retval;
310   va_list args;
311   va_start (args, fmt);
312
313   retval = vfprintf (stderr, GETTEXT (fmt), args);
314
315   va_end (args);
316   /* fflush (stderr); */
317   return retval;
318 }
319
320 /* exactly equivalent to fprintf (stdout, fmt, ...) except that it calls
321    GETTEXT on the format string. */
322
323 int
324 stdout_out (CONST char *fmt, ...)
325 {
326   int retval;
327   va_list args;
328   va_start (args, fmt);
329
330   retval = vfprintf (stdout, GETTEXT (fmt), args);
331
332   va_end (args);
333   return retval;
334 }
335
336 #ifdef SIGDANGER
337
338 /* Handler for SIGDANGER.  */
339 SIGTYPE
340 memory_warning_signal (int sig)
341 {
342   /* #### bad bad bad; this function shouldn't do anything except
343      set a flag, or weird corruption could happen. */
344   signal (sig, memory_warning_signal);
345
346   malloc_warning
347     (GETTEXT ("Operating system warns that virtual memory is running low.\n"));
348
349   /* It might be unsafe to call do_auto_save now.  */
350   force_auto_save_soon ();
351 }
352 #endif /* SIGDANGER */
353 \f
354 /* Code for dealing with Lisp access to the Unix command line */
355
356 static Lisp_Object
357 make_arg_list_1 (int argc, char **argv, int skip_args)
358 {
359   Lisp_Object result = Qnil;
360   REGISTER int i;
361
362   for (i = argc - 1; i >= 0; i--)
363     {
364       if (i == 0 || i > skip_args)
365         {
366 #ifdef WINDOWSNT
367           if (i == 0)
368             {
369               /* Do not trust to what crt0 has stuffed into argv[0] */
370               char full_exe_path [MAX_PATH];
371               GetModuleFileName (NULL, full_exe_path, MAX_PATH);
372               result = Fcons (build_ext_string (full_exe_path, FORMAT_FILENAME),
373                               result);
374 #if defined(HAVE_SHLIB)
375               (void)dll_init(full_exe_path);
376 #endif
377             }
378           else
379 #endif
380             result = Fcons (build_ext_string (argv [i], FORMAT_FILENAME), result);
381         }
382     }
383   return result;
384 }
385
386 Lisp_Object
387 make_arg_list (int argc, char **argv)
388 {
389   return make_arg_list_1 (argc, argv, 0);
390 }
391
392 /* Calling functions are also responsible for calling free_argc_argv
393    when they are done with the generated list. */
394 void
395 make_argc_argv (Lisp_Object argv_list, int *argc, char ***argv)
396 {
397   Lisp_Object next;
398   int n = XINT (Flength (argv_list));
399   REGISTER int i;
400   *argv = (char**) xmalloc ((n+1) * sizeof (char*));
401
402   for (i = 0, next = argv_list; i < n; i++, next = XCDR (next))
403     {
404       CONST char *temp;
405       CHECK_STRING (XCAR (next));
406
407       GET_C_STRING_EXT_DATA_ALLOCA (XCAR (next), FORMAT_OS, temp);
408       (*argv) [i] = xstrdup (temp);
409     }
410   (*argv) [n] = 0;
411   *argc = i;
412 }
413
414 void
415 free_argc_argv (char **argv)
416 {
417   int elt = 0;
418
419   while (argv[elt])
420     {
421       xfree (argv[elt]);
422       elt++;
423     }
424   xfree (argv);
425 }
426
427 static void
428 init_cmdargs (int argc, char **argv, int skip_args)
429 {
430   initial_argv = argv;
431   initial_argc = argc;
432
433   Vcommand_line_args = make_arg_list_1 (argc, argv, skip_args);
434 }
435
436 DEFUN ("invocation-name", Finvocation_name, 0, 0, 0, /*
437 Return the program name that was used to run XEmacs.
438 Any directory names are omitted.
439 */
440        ())
441 {
442   return Fcopy_sequence (Vinvocation_name);
443 }
444
445 DEFUN ("invocation-directory", Finvocation_directory, 0, 0, 0, /*
446 Return the directory name in which the Emacs executable was located.
447 */
448        ())
449 {
450   return Fcopy_sequence (Vinvocation_directory);
451 }
452
453 \f
454 #ifdef I18N4
455                                 /* #### - don't know why I18N4 on SunOS/JLE
456                                    can't deal with this.  It's a potential
457                                    bug that needs to be looked at. */
458 # undef RUN_TIME_REMAP
459 #endif
460
461 #if defined (MULE) && defined (MSDOS) && defined (EMX)
462 /* Setup all of files be input/output'ed with binary translation mode. */
463 asm ("  .text");
464 asm ("L_setbinmode:");
465 asm ("  movl    $1, __fmode_bin");
466 asm ("  ret");
467 asm ("  .stabs  \"___CTOR_LIST__\", 23, 0, 0, L_setbinmode");
468 #endif
469
470 /* Test whether the next argument in ARGV matches SSTR or a prefix of
471    LSTR (at least MINLEN characters).  If so, then if VALPTR is non-null
472    (the argument is supposed to have a value) store in *VALPTR either
473    the next argument or the portion of this one after the equal sign.
474    ARGV is read starting at position *SKIPPTR; this index is advanced
475    by the number of arguments used.
476
477    Too bad we can't just use getopt for all of this, but we don't have
478    enough information to do it right.  */
479
480 static int
481 argmatch (char **argv, int argc, char *sstr, char *lstr,
482           int minlen, char **valptr, int *skipptr)
483 {
484   char *p = NULL;
485   int arglen;
486   char *arg;
487
488   /* Don't access argv[argc]; give up in advance.  */
489   if (argc <= *skipptr + 1)
490     return 0;
491
492   arg = argv[*skipptr+1];
493   if (arg == NULL)
494     return 0;
495   if (strcmp (arg, sstr) == 0)
496     {
497       if (valptr != NULL)
498         {
499           *valptr = argv[*skipptr+2];
500           *skipptr += 2;
501         }
502       else
503         *skipptr += 1;
504       return 1;
505     }
506   arglen = (valptr != NULL && (p = strchr (arg, '=')) != NULL
507             ? p - arg : strlen (arg));
508   if (lstr == 0 || arglen < minlen || strncmp (arg, lstr, arglen) != 0)
509     return 0;
510   else if (valptr == NULL)
511     {
512       *skipptr += 1;
513       return 1;
514     }
515   else if (p != NULL)
516     {
517       *valptr = p+1;
518       *skipptr += 1;
519       return 1;
520     }
521   else if (argv[*skipptr+2] != NULL)
522     {
523       *valptr = argv[*skipptr+2];
524       *skipptr += 2;
525       return 1;
526     }
527   else
528     {
529       return 0;
530     }
531 }
532
533 /* Make stack traces always identify version + configuration */
534 #define main_1 STACK_TRACE_EYE_CATCHER
535
536 /* This function is not static, so that the compiler is less likely to
537    inline it, which would make it not show up in stack traces.  */
538 DECLARE_DOESNT_RETURN (main_1 (int, char **, char **, int));
539 DOESNT_RETURN
540 main_1 (int argc, char **argv, char **envp, int restart)
541 {
542   char stack_bottom_variable;
543   int skip_args = 0;
544   Lisp_Object load_me;
545   int inhibit_window_system;
546 #ifdef NeXT
547   extern int malloc_cookie;
548 #endif
549
550 #if !defined(SYSTEM_MALLOC) && !defined(HAVE_LIBMCHECK)
551   /* Make sure that any libraries we link against haven't installed a
552      hook for a gmalloc of a potentially incompatible version. */
553   /* If we're using libmcheck, the hooks have already been initialized, */
554   /* don't touch them. -slb */
555   __malloc_hook = NULL;
556   __realloc_hook = NULL;
557   __free_hook = NULL;
558 #endif /* not SYSTEM_MALLOC */
559
560   noninteractive = 0;
561
562 #ifdef NeXT
563   /* 19-Jun-1995 -baw
564    * NeXT secret magic, ripped from Emacs-for-NS by Carl Edman
565    * <cedman@princeton.edu>.  Note that even Carl doesn't know what this
566    * does; it was provided by NeXT, and it presumable makes NS's mallocator
567    * work with dumping.  But malloc_jumpstart() and malloc_freezedry() in
568    * unexnext.c are both completely undocumented, even in NS header files!
569    * But hey, it solves all NS related memory problems, so who's
570    * complaining? */
571   if (initialized && malloc_jumpstart (malloc_cookie) != 0)
572     fprintf (stderr, "malloc jumpstart failed!\n");
573 #endif /* NeXT */
574
575   /*
576 #if defined (GNU_MALLOC) && \
577     defined (ERROR_CHECK_MALLOC) && \
578     !defined (HAVE_LIBMCHECK)
579   */
580 #if defined(LOSING_GCC_DESTRUCTOR_FREE_BUG)
581   /* Prior to XEmacs 21, this was `#if 0'ed out.  */
582   /* I'm enabling this because it is the only reliable way I've found to */
583   /* prevent a very annoying problem where GCC will attempt to free(3) */
584   /* memory at exit() and cause a coredump. */
585   init_free_hook ();
586 #endif
587
588   sort_args (argc, argv);
589
590   /* Map in shared memory, if we are using that.  */
591 #ifdef HAVE_SHM
592   if (argmatch (argv, argc, "-nl", "--no-shared-memory", 6, NULL, &skip_args))
593     {
594       map_in_data (0);
595       /* The shared memory was just restored, which clobbered this.  */
596       skip_args = 1;
597     }
598   else
599     {
600       map_in_data (1);
601       /* The shared memory was just restored, which clobbered this.  */
602       skip_args = 0;
603     }
604 #endif /* HAVE_SHM */
605
606 #if (defined (MSDOS) && defined (EMX)) || defined (WIN32) || defined (_SCO_DS)
607   environ = envp;
608 #endif
609
610   /* Record (approximately) where the stack begins.  */
611   stack_bottom = &stack_bottom_variable;
612
613 #ifdef USG_SHARED_LIBRARIES
614   if (bss_end)
615     brk ((void *) bss_end);
616 #endif
617
618   clearerr (stdin);
619
620 #ifdef APOLLO
621 #ifndef APOLLO_SR10
622   /* If USE_DOMAIN_ACLS environment variable exists,
623      use ACLs rather than UNIX modes. */
624   if (egetenv ("USE_DOMAIN_ACLS"))
625     default_acl (USE_DEFACL);
626 #endif
627 #endif /* APOLLO */
628
629 #if defined (HAVE_MMAP) && defined (REL_ALLOC)
630   /* ralloc can only be used if using the GNU memory allocator. */
631   init_ralloc ();
632 #elif defined (REL_ALLOC) && !defined(DOUG_LEA_MALLOC)
633   if (initialized)
634     init_ralloc();
635 #endif
636
637 #ifdef HAVE_SOCKS
638   if (initialized)
639     SOCKSinit (argv[0]);
640 #endif /* HAVE_SOCKS */
641
642 #ifndef SYSTEM_MALLOC
643   if (!initialized)
644     /* Arrange to get warning messages as memory fills up.  */
645     memory_warnings (0, malloc_warning);
646 #endif  /* not SYSTEM_MALLOC */
647
648 #ifdef MSDOS
649   /* We do all file input/output as binary files.  When we need to translate
650      newlines, we do that manually.  */
651   _fmode = O_BINARY;
652   (stdin) ->_flag &= ~_IOTEXT;
653   (stdout)->_flag &= ~_IOTEXT;
654   (stderr)->_flag &= ~_IOTEXT;
655 #endif /* MSDOS */
656
657 #ifdef SET_EMACS_PRIORITY
658   if (emacs_priority != 0)
659     nice (-emacs_priority);
660   setuid (getuid ());
661 #endif /* SET_EMACS_PRIORITY */
662
663 #ifdef EXTRA_INITIALIZE
664   EXTRA_INITIALIZE;
665 #endif
666
667 #ifdef HAVE_WINDOW_SYSTEM
668   inhibit_window_system = 0;
669 #else
670   inhibit_window_system = 1;
671 #endif
672
673   /* Handle the -t switch, which specifies filename to use as terminal */
674   {
675     char *term;
676     if (argmatch (argv, argc, "-t", "--terminal", 4, &term, &skip_args))
677       {
678         close (0);
679         close (1);
680         if (open (term, O_RDWR | OPEN_BINARY, 2) < 0)
681           fatal ("%s: %s", term, strerror (errno));
682         dup (0);
683         if (! isatty (0))
684           fatal ("%s: not a tty", term);
685
686 #if 0
687         stderr_out ("Using %s", ttyname (0));
688 #endif
689         stderr_out ("Using %s", term);
690         inhibit_window_system = 1;      /* -t => -nw */
691       }
692   }
693
694   /* Handle -nw switch */
695   if (argmatch (argv, argc, "-nw", "--no-windows", 6, NULL, &skip_args))
696     inhibit_window_system = 1;
697
698   /* Handle the -batch switch, which means don't do interactive display.  */
699   if (argmatch (argv, argc, "-batch", "--batch", 5, NULL, &skip_args))
700     {
701 #if 0 /* I don't think this is correct. */
702       inhibit_autoloads = 1;
703 #endif
704       noninteractive = 1;
705     }
706
707   if (argmatch (argv, argc, "-debug-paths", "--debug-paths",
708                 11, NULL, &skip_args))
709       debug_paths = 1;
710
711   /* Partially handle -no-autoloads, -no-early-packages and -vanilla.  Packages */
712   /* are searched prior to the rest of the command line being parsed in */
713   /* startup.el */
714   if (argmatch (argv, argc, "-no-early-packages", "--no-early-packages",
715                 6, NULL, &skip_args))
716     {
717       inhibit_early_packages = 1;
718       skip_args--;
719     }
720 #ifdef HAVE_SHLIB
721   if (argmatch (argv, argc, "-no-site-modules", "--no-site-modules",
722                 9, NULL, &skip_args))
723     {
724       inhibit_site_modules = 1;
725       skip_args--;
726     }
727 #else
728   inhibit_site_modules = 1;
729 #endif
730   if (argmatch (argv, argc, "-vanilla", "--vanilla",
731                 7, NULL, &skip_args))
732     {
733       inhibit_early_packages = 1;
734       skip_args--;
735     }
736
737   if (argmatch (argv, argc, "-no-autoloads", "--no-autoloads",
738                 7, NULL, &skip_args))
739     {
740       /* Inhibit everything */
741       inhibit_autoloads = 1;
742       skip_args--;
743     }
744
745   if (argmatch (argv, argc, "-debug-paths", "--debug-paths",
746                 6, NULL, &skip_args))
747     {
748       debug_paths = 1;
749       skip_args--;
750     }
751
752
753   /* Partially handle the -version and -help switches: they imply -batch,
754      but are not removed from the list. */
755   if (argmatch (argv, argc, "-help", "--help",   3, NULL, &skip_args))
756     noninteractive = 1, skip_args--;
757
758   if (argmatch (argv, argc, "-version", "--version", 3, NULL, &skip_args) ||
759       argmatch (argv, argc, "-V",    0,              2, NULL, &skip_args))
760       noninteractive = 1, skip_args--;
761
762   /* Now, figure out which type of console is our first console. */
763
764   display_arg = 0;
765
766   if (noninteractive)
767     display_use = "stream";
768   else
769     display_use = "tty";
770
771 #ifndef HAVE_TTY
772   if (inhibit_window_system)
773     fatal ("Sorry, this XEmacs was not compiled with TTY support");
774 #endif
775
776 #ifdef HAVE_WINDOW_SYSTEM
777   /* Stupid kludge to catch command-line display spec.  We can't
778      handle this argument entirely in window-system-dependent code
779      because we don't even know which window-system-dependent code
780      to run until we've recognized this argument.  */
781   if (!inhibit_window_system && !noninteractive)
782     {
783 #ifdef HAVE_X_WINDOWS
784       char *dpy = 0;
785       int count_before = skip_args;
786
787       if (argmatch (argv, argc, "-d", "--display", 3, &dpy, &skip_args) ||
788           argmatch (argv, argc, "-display", 0,     3, &dpy, &skip_args))
789         {
790           display_arg = 1;
791           display_use = "x";
792         }
793       /* If we have the form --display=NAME,
794          convert it into  -d name.
795          This requires inserting a new element into argv.  */
796       if (dpy != 0 && skip_args - count_before == 1)
797         {
798           char **new = (char **) xmalloc (sizeof (char *) * (argc + 2));
799           int j;
800
801           for (j = 0; j < count_before + 1; j++)
802             new[j] = argv[j];
803           new[count_before + 1] = "-d";
804           new[count_before + 2] = dpy;
805           for (j = count_before + 2; j <argc; j++)
806             new[j + 1] = argv[j];
807           argv = new;
808           argc++;
809         }
810       /* Change --display to -d, when its arg is separate.  */
811       else if (dpy != 0 && skip_args > count_before
812                && argv[count_before + 1][1] == '-')
813         argv[count_before + 1] = "-d";
814
815       /* Don't actually discard this arg.  */
816       skip_args = count_before;
817
818       /* If there is a non-empty environment var DISPLAY, set
819          `display_use', but not `display_arg', which is only to be set
820          if the display was specified on the command line. */
821       if ((dpy = getenv ("DISPLAY")) && dpy[0])
822         display_use = "x";
823
824 #endif /* HAVE_X_WINDOWS */
825 #ifdef HAVE_MS_WINDOWS
826       if (strcmp(display_use, "x") != 0)
827         display_use = "mswindows";
828 #endif /* HAVE_MS_WINDOWS */
829     }
830 #endif /* HAVE_WINDOW_SYSTEM */
831
832   noninteractive1 = noninteractive;
833
834   /****** Now initialize everything *******/
835
836   /* First, do really basic environment initialization -- catching signals
837      and the like.  These functions have no dependence on any part of
838      the Lisp engine and need to be done both at dump time and at run time. */
839
840   init_signals_very_early ();
841   init_data_very_early (); /* Catch math errors. */
842 #ifdef LISP_FLOAT_TYPE
843   init_floatfns_very_early (); /* Catch floating-point math errors. */
844 #endif
845   init_process_times_very_early (); /* Initialize our process timers.
846                                        As early as possible, of course,
847                                        so we can be fairly accurate. */
848   init_intl_very_early (); /* set up the locale and domain for gettext and
849                               such. */
850
851   /* Now initialize the Lisp engine and the like.  Done only during
852      dumping.  No dependence on anything that may be in the user's
853      environment when the dumped XEmacs is run.
854
855      We try to do things in an order that minimizes the non-obvious
856      dependencies between functions. */
857
858   if (!initialized)
859     {
860       /* Initialize things so that new Lisp objects
861          can be created and objects can be staticpro'd.
862          Must be basically the very first thing done
863          because pretty much all of the initialization
864          routines below create new objects. */
865       init_alloc_once_early ();
866
867       /* Initialize Qnil, Qt, Qunbound, and the
868          obarray.  After this, symbols can be
869          interned.  This depends on init_alloc_once(). */
870       init_symbols_once_early ();
871
872       /* Declare the basic symbols pertaining to errors,
873          So that deferror() can be called. */
874       init_errors_once_early ();
875
876       /* Make sure that opaque pointers can be created. */
877       init_opaque_once_early ();
878
879       /* Now declare all the symbols and define all the Lisp primitives.
880
881          The *only* thing that the syms_of_*() functions are allowed to do
882          is call one of the following three functions:
883
884          defsymbol()
885          defsubr() (i.e. DEFSUBR)
886          deferror()
887          defkeyword()
888
889          Order does not matter in these functions.
890          */
891
892       syms_of_abbrev ();
893       syms_of_alloc ();
894 #ifdef HAVE_X_WINDOWS
895       syms_of_balloon_x ();
896 #endif
897       syms_of_buffer ();
898       syms_of_bytecode ();
899       syms_of_callint ();
900       syms_of_callproc ();
901       syms_of_casefiddle ();
902       syms_of_casetab ();
903       syms_of_chartab ();
904       syms_of_cmdloop ();
905       syms_of_cmds ();
906       syms_of_console ();
907       syms_of_data ();
908 #ifdef DEBUG_XEMACS
909       syms_of_debug ();
910 #endif /* DEBUG_XEMACS */
911       syms_of_device ();
912 #ifdef HAVE_DIALOGS
913       syms_of_dialog ();
914 #endif
915       syms_of_dired ();
916       syms_of_doc ();
917       syms_of_editfns ();
918       syms_of_elhash ();
919       syms_of_emacs ();
920       syms_of_eval ();
921 #ifdef HAVE_X_WINDOWS
922       syms_of_event_Xt ();
923 #endif
924 #ifdef HAVE_DRAGNDROP
925       syms_of_dragdrop ();
926 #endif
927       syms_of_event_stream ();
928       syms_of_events ();
929       syms_of_extents ();
930       syms_of_faces ();
931       syms_of_fileio ();
932 #ifdef CLASH_DETECTION
933       syms_of_filelock ();
934 #endif /* CLASH_DETECTION */
935       syms_of_floatfns ();
936       syms_of_fns ();
937       syms_of_font_lock ();
938       syms_of_frame ();
939       syms_of_general ();
940       syms_of_glyphs ();
941       syms_of_glyphs_eimage ();
942       syms_of_glyphs_widget ();
943 #if defined (HAVE_MENUBARS) || defined (HAVE_SCROLLBARS) || defined (HAVE_DIALOGS) || defined (HAVE_TOOLBARS)
944       syms_of_gui ();
945 #endif
946       syms_of_indent ();
947       syms_of_intl ();
948       syms_of_keymap ();
949       syms_of_lread ();
950       syms_of_macros ();
951       syms_of_marker ();
952       syms_of_md5 ();
953 #ifdef HAVE_DATABASE
954       syms_of_database ();
955 #endif
956 #ifdef HAVE_MENUBARS
957       syms_of_menubar ();
958 #endif
959       syms_of_minibuf ();
960 #ifdef HAVE_SHLIB
961       syms_of_module ();
962 #endif
963       syms_of_objects ();
964       syms_of_print ();
965 #if !defined (NO_SUBPROCESSES)
966       syms_of_process ();
967 #ifdef HAVE_WIN32_PROCESSES
968       syms_of_process_nt ();
969 #endif
970 #endif
971       syms_of_profile ();
972 #if defined (HAVE_MMAP) && defined (REL_ALLOC) && !defined(DOUG_LEA_MALLOC)
973       syms_of_ralloc ();
974 #endif /* HAVE_MMAP && REL_ALLOC */
975       syms_of_rangetab ();
976       syms_of_redisplay ();
977       syms_of_search ();
978       syms_of_select ();
979       syms_of_signal ();
980       syms_of_sound ();
981       syms_of_specifier ();
982       syms_of_symbols ();
983       syms_of_syntax ();
984 #ifdef HAVE_SCROLLBARS
985       syms_of_scrollbar ();
986 #endif
987 #ifdef HAVE_TOOLBARS
988       syms_of_toolbar ();
989 #endif
990       syms_of_undo ();
991       syms_of_widget ();
992       syms_of_window ();
993
994 #ifdef HAVE_TTY
995       syms_of_console_tty ();
996       syms_of_device_tty ();
997       syms_of_objects_tty ();
998 #endif
999
1000 #ifdef HAVE_X_WINDOWS
1001       syms_of_device_x ();
1002 #ifdef HAVE_DIALOGS
1003       syms_of_dialog_x ();
1004 #endif
1005       syms_of_frame_x ();
1006       syms_of_glyphs_x ();
1007       syms_of_objects_x ();
1008 #ifdef HAVE_MENUBARS
1009       syms_of_menubar_x ();
1010 #endif
1011       syms_of_xselect ();
1012 #if defined (HAVE_MENUBARS) || defined (HAVE_SCROLLBARS) || defined (HAVE_DIALOGS) || defined (HAVE_TOOLBARS)
1013       syms_of_gui_x ();
1014 #endif
1015 #endif /* HAVE_X_WINDOWS */
1016
1017 #ifdef HAVE_MS_WINDOWS
1018       syms_of_console_mswindows ();
1019       syms_of_device_mswindows ();
1020       syms_of_frame_mswindows ();
1021       syms_of_objects_mswindows ();
1022       syms_of_select_mswindows ();
1023       syms_of_glyphs_mswindows ();
1024 #ifdef HAVE_MENUBARS
1025       syms_of_menubar_mswindows ();
1026 #endif
1027 #ifdef HAVE_SCROLLBARS
1028       syms_of_scrollbar_mswindows ();
1029 #endif
1030 #ifdef HAVE_MSW_C_DIRED
1031       syms_of_dired_mswindows ();
1032 #endif
1033 #ifdef WINDOWSNT
1034       syms_of_ntproc ();
1035 #endif
1036 #endif  /* HAVE_MS_WINDOWS */
1037
1038 #ifdef MULE
1039       syms_of_mule ();
1040       syms_of_mule_ccl ();
1041       syms_of_mule_charset ();
1042 #endif
1043 #ifdef FILE_CODING
1044       syms_of_file_coding ();
1045 #endif
1046 #ifdef MULE
1047 #ifdef HAVE_WNN
1048       syms_of_mule_wnn ();
1049 #endif
1050 #ifdef HAVE_CANNA
1051       syms_of_mule_canna ();
1052 #endif /* HAVE_CANNA */
1053 #endif /* MULE */
1054
1055 #ifdef SYMS_SYSTEM
1056       SYMS_SYSTEM;
1057 #endif
1058
1059 #ifdef SYMS_MACHINE
1060       SYMS_MACHINE;
1061 #endif
1062
1063       /*
1064 #if defined (GNU_MALLOC) && \
1065     defined (ERROR_CHECK_MALLOC) && \
1066     !defined (HAVE_LIBMCHECK)
1067       */
1068       /* Prior to XEmacs 21, this was `#if 0'ed out. -slb */
1069 #if defined (LOSING_GCC_DESTRUCTOR_FREE_BUG)
1070       syms_of_free_hook ();
1071 #endif
1072
1073 #ifdef TOOLTALK
1074       syms_of_tooltalk ();
1075 #endif
1076
1077 #ifdef SUNPRO
1078       syms_of_sunpro ();
1079 #endif
1080
1081 #ifdef HAVE_LDAP
1082       syms_of_eldap ();
1083 #endif
1084
1085       /* Now create the subtypes for the types that have them.
1086          We do this before the vars_*() because more symbols
1087          may get initialized here. */
1088
1089       /* Now initialize the console types and associated symbols.
1090          Other than the first function below, the functions may
1091          make exactly the following function/macro calls:
1092
1093          INITIALIZE_CONSOLE_TYPE()
1094          CONSOLE_HAS_METHOD()
1095
1096          For any given console type, the former macro must be called
1097          before the any calls to the latter macro. */
1098
1099       console_type_create ();
1100
1101       console_type_create_stream ();
1102
1103 #ifdef HAVE_TTY
1104       console_type_create_tty ();
1105       console_type_create_device_tty ();
1106       console_type_create_frame_tty ();
1107       console_type_create_objects_tty ();
1108       console_type_create_redisplay_tty ();
1109 #endif
1110
1111 #ifdef HAVE_X_WINDOWS
1112       console_type_create_x ();
1113       console_type_create_device_x ();
1114       console_type_create_frame_x ();
1115       console_type_create_glyphs_x ();
1116       console_type_create_select_x ();
1117 #ifdef HAVE_MENUBARS
1118       console_type_create_menubar_x ();
1119 #endif
1120       console_type_create_objects_x ();
1121       console_type_create_redisplay_x ();
1122 #ifdef HAVE_SCROLLBARS
1123       console_type_create_scrollbar_x ();
1124 #endif
1125 #ifdef HAVE_TOOLBARS
1126       console_type_create_toolbar_x ();
1127 #endif
1128 #ifdef HAVE_DIALOGS
1129       console_type_create_dialog_x ();
1130 #endif
1131 #endif /* HAVE_X_WINDOWS */
1132
1133 #ifdef HAVE_MS_WINDOWS
1134       console_type_create_mswindows ();
1135       console_type_create_device_mswindows ();
1136       console_type_create_frame_mswindows ();
1137       console_type_create_objects_mswindows ();
1138       console_type_create_redisplay_mswindows ();
1139       console_type_create_glyphs_mswindows ();
1140       console_type_create_select_mswindows ();
1141 # ifdef HAVE_SCROLLBARS
1142       console_type_create_scrollbar_mswindows ();
1143 # endif
1144 #ifdef HAVE_MENUBARS
1145       console_type_create_menubar_mswindows ();
1146 #endif
1147 #ifdef HAVE_TOOLBARS
1148       console_type_create_toolbar_mswindows ();
1149 #endif
1150 #ifdef HAVE_DIALOGS
1151       console_type_create_dialog_mswindows ();
1152 #endif
1153 #endif
1154
1155       /* Now initialize the specifier types and associated symbols.
1156          Other than the first function below, the functions may
1157          make exactly the following function/macro calls:
1158
1159          INITIALIZE_SPECIFIER_TYPE()
1160          SPECIFIER_HAS_METHOD()
1161
1162          For any given specifier type, the former macro must be called
1163          before the any calls to the latter macro. */
1164
1165       specifier_type_create ();
1166
1167       specifier_type_create_image ();
1168       specifier_type_create_objects ();
1169 #ifdef HAVE_TOOLBARS
1170       specifier_type_create_toolbar ();
1171 #endif
1172
1173       /* Now initialize the structure types and associated symbols.
1174          Other than the first function below, the functions may
1175          make exactly the following function/macro calls:
1176
1177          define_structure_type()
1178          define_structure_type_keyword()
1179
1180          */
1181
1182       structure_type_create ();
1183
1184       structure_type_create_chartab ();
1185       structure_type_create_faces ();
1186       structure_type_create_rangetab ();
1187       structure_type_create_hash_table ();
1188
1189       /* Now initialize the image instantiator formats and associated symbols.
1190          Other than the first function below, the functions may
1191          make exactly the following function/macro calls:
1192
1193          INITIALIZE_IMAGE_INSTANTIATOR_FORMAT()
1194          IIFORMAT_HAS_METHOD()
1195          IIFORMAT_VALID_KEYWORD()
1196
1197          For any given image instantiator format, the first macro must be
1198          called before the any calls to the other macros. */
1199
1200       image_instantiator_format_create ();
1201       image_instantiator_format_create_glyphs_eimage ();
1202       image_instantiator_format_create_glyphs_widget ();
1203 #ifdef HAVE_TTY
1204       image_instantiator_format_create_glyphs_tty ();
1205 #endif
1206 #ifdef HAVE_X_WINDOWS
1207       image_instantiator_format_create_glyphs_x ();
1208 #endif /* HAVE_X_WINDOWS */
1209 #ifdef HAVE_MS_WINDOWS
1210       image_instantiator_format_create_glyphs_mswindows ();
1211 #endif /* HAVE_MSWINDOWS_WINDOWS */
1212
1213       /* Now initialize the lstream types and associated symbols.
1214          Other than the first function below, the functions may
1215          make exactly the following function/macro calls:
1216
1217          LSTREAM_HAS_METHOD()
1218
1219          */
1220
1221       lstream_type_create ();
1222 #ifdef FILE_CODING
1223       lstream_type_create_file_coding ();
1224 #endif
1225 #if defined (HAVE_MS_WINDOWS) && !defined(HAVE_MSG_SELECT)
1226       lstream_type_create_mswindows_selectable ();
1227 #endif
1228
1229       /* Initialize processes implementation.
1230          The functions may make exactly the following function/macro calls:
1231
1232          PROCESS_HAS_METHOD()
1233       */
1234 #ifdef HAVE_UNIX_PROCESSES
1235       process_type_create_unix ();
1236 #endif
1237 #ifdef HAVE_WIN32_PROCESSES
1238       process_type_create_nt ();
1239 #endif
1240
1241       /* Now initialize most variables.
1242
1243          These functions may do exactly the following:
1244
1245          DEFVAR_INT()
1246          DEFVAR_LISP()
1247          DEFVAR_BOOL()
1248          DEFER_GETTEXT()
1249          Dynarr_*()
1250          Blocktype_*()
1251          staticpro()
1252          Fprovide(symbol)
1253          intern()
1254          pure_put()
1255          xmalloc()
1256          defsymbol(), if it's absolutely necessary and you're sure that
1257            the symbol isn't referenced anywhere else in the initialization
1258            code
1259          Fset() on a symbol that is unbound
1260          assigning a symbol or constant value to a variable
1261          using a global variable that has been initialized
1262            earlier on in the same function
1263
1264          Any of the object-creating functions on alloc.c: e.g.
1265
1266          make_pure_*()
1267          Fpurecopy()
1268          make_string()
1269          build_string()
1270          make_vector()
1271          make_int()
1272          make_extent()
1273          alloc_lcrecord()
1274          Fcons()
1275          listN()
1276          make_opaque_ptr()
1277          make_opaque_long()
1278
1279          perhaps a few others.
1280        */
1281
1282       /* Now allow Fprovide() statements to be made. */
1283       init_provide_once ();
1284
1285       vars_of_abbrev ();
1286       vars_of_alloc ();
1287 #ifdef HAVE_X_WINDOWS
1288       vars_of_balloon_x ();
1289 #endif
1290       vars_of_buffer ();
1291       vars_of_bytecode ();
1292       vars_of_callint ();
1293       vars_of_callproc ();
1294       vars_of_chartab ();
1295       vars_of_cmdloop ();
1296       vars_of_cmds ();
1297       vars_of_console ();
1298       vars_of_data ();
1299 #ifdef DEBUG_XEMACS
1300       vars_of_debug ();
1301 #endif
1302       vars_of_console_stream ();
1303       vars_of_device ();
1304 #ifdef HAVE_DIALOGS
1305       vars_of_dialog ();
1306 #endif
1307       vars_of_dired ();
1308       vars_of_doc ();
1309 #ifdef HAVE_DRAGNDROP
1310       vars_of_dragdrop ();
1311 #endif
1312       vars_of_editfns ();
1313       vars_of_elhash ();
1314       vars_of_emacs ();
1315       vars_of_eval ();
1316
1317 #ifdef HAVE_X_WINDOWS
1318       vars_of_event_Xt ();
1319 #endif
1320 #if defined(HAVE_TTY) && (defined (DEBUG_TTY_EVENT_STREAM) || !defined (HAVE_X_WINDOWS))
1321       vars_of_event_tty ();
1322 #endif
1323 #ifdef HAVE_MS_WINDOWS
1324       vars_of_event_mswindows ();
1325 #endif
1326       vars_of_event_stream ();
1327
1328       vars_of_events ();
1329       vars_of_extents ();
1330       vars_of_faces ();
1331       vars_of_fileio ();
1332       vars_of_floatfns ();
1333       vars_of_font_lock ();
1334       vars_of_frame ();
1335       vars_of_glyphs ();
1336       vars_of_glyphs_eimage ();
1337       vars_of_glyphs_widget ();
1338 #if defined (HAVE_MENUBARS) || defined (HAVE_SCROLLBARS) || defined (HAVE_DIALOGS) || defined (HAVE_TOOLBARS)
1339       vars_of_gui ();
1340 #endif
1341       vars_of_indent ();
1342       vars_of_insdel ();
1343       vars_of_intl ();
1344 #ifdef HAVE_XIM
1345 #ifdef XIM_MOTIF
1346       vars_of_input_method_motif ();
1347 #else /* XIM_XLIB */
1348       vars_of_input_method_xlib ();
1349 #endif
1350 #endif /* HAVE_XIM */
1351       vars_of_keymap ();
1352       vars_of_lread ();
1353       vars_of_lstream ();
1354       vars_of_macros ();
1355       vars_of_md5 ();
1356 #ifdef HAVE_DATABASE
1357       vars_of_database ();
1358 #endif
1359 #ifdef HAVE_MENUBARS
1360       vars_of_menubar ();
1361 #endif
1362       vars_of_minibuf ();
1363 #ifdef HAVE_SHLIB
1364       vars_of_module ();
1365 #endif
1366 #ifdef WINDOWSNT
1367       vars_of_ntproc ();
1368 #endif
1369       vars_of_objects ();
1370       vars_of_print ();
1371
1372 #ifndef NO_SUBPROCESSES
1373       vars_of_process ();
1374 #ifdef HAVE_UNIX_PROCESSES
1375       vars_of_process_unix ();
1376 #endif
1377 #ifdef HAVE_WIN32_PROCESSES
1378       vars_of_process_nt ();
1379 #endif
1380 #endif
1381
1382       vars_of_profile ();
1383 #if defined (HAVE_MMAP) && defined (REL_ALLOC) && !defined(DOUG_LEA_MALLOC)
1384       vars_of_ralloc ();
1385 #endif /* HAVE_MMAP && REL_ALLOC */
1386       vars_of_redisplay ();
1387 #ifdef HAVE_SCROLLBARS
1388       vars_of_scrollbar ();
1389 #endif
1390       vars_of_search ();
1391       vars_of_select ();
1392       vars_of_sound ();
1393       vars_of_specifier ();
1394       vars_of_symbols ();
1395       vars_of_syntax ();
1396 #ifdef HAVE_TOOLBARS
1397       vars_of_toolbar ();
1398 #endif
1399       vars_of_undo ();
1400       vars_of_window ();
1401
1402 #ifdef HAVE_TTY
1403       vars_of_console_tty ();
1404       vars_of_frame_tty ();
1405       vars_of_objects_tty ();
1406 #endif
1407
1408 #ifdef HAVE_X_WINDOWS
1409       vars_of_device_x ();
1410 #ifdef HAVE_DIALOGS
1411       vars_of_dialog_x ();
1412 #endif
1413       vars_of_frame_x ();
1414       vars_of_glyphs_x ();
1415 #ifdef HAVE_MENUBARS
1416       vars_of_menubar_x ();
1417 #endif
1418       vars_of_objects_x ();
1419       vars_of_xselect ();
1420 #ifdef HAVE_SCROLLBARS
1421       vars_of_scrollbar_x ();
1422 #endif
1423 #if defined (HAVE_MENUBARS) || defined (HAVE_SCROLLBARS) || defined (HAVE_DIALOGS) || defined (HAVE_TOOLBARS)
1424       vars_of_gui_x ();
1425 #endif
1426 #endif
1427
1428 #ifdef HAVE_MS_WINDOWS
1429       vars_of_device_mswindows ();
1430       vars_of_console_mswindows ();
1431       vars_of_frame_mswindows ();
1432       vars_of_objects_mswindows ();
1433       vars_of_select_mswindows ();
1434       vars_of_glyphs_mswindows ();
1435 #ifdef HAVE_SCROLLBARS
1436       vars_of_scrollbar_mswindows ();
1437 #endif
1438 #ifdef HAVE_MENUBARS
1439       vars_of_menubar_mswindows ();
1440 #endif
1441 #ifdef HAVE_MSW_C_DIRED
1442       vars_of_dired_mswindows ();
1443 #endif
1444 #ifdef HAVE_DIALOGS
1445       vars_of_dialog_mswindows ();
1446 #endif
1447 #endif  /* HAVE_MS_WINDOWS */
1448
1449 #ifdef MULE
1450       vars_of_mule ();
1451       vars_of_mule_ccl ();
1452       vars_of_mule_charset ();
1453 #endif
1454 #ifdef FILE_CODING
1455       vars_of_file_coding ();
1456 #endif
1457 #ifdef MULE
1458 #ifdef HAVE_WNN
1459       vars_of_mule_wnn ();
1460 #endif
1461 #ifdef HAVE_CANNA
1462       vars_of_mule_canna ();
1463 #endif /* HAVE_CANNA */
1464 #endif /* MULE */
1465
1466 #ifdef TOOLTALK
1467       vars_of_tooltalk ();
1468 #endif
1469
1470 #ifdef SUNPRO
1471       vars_of_sunpro ();
1472 #endif
1473
1474 #ifdef HAVE_LDAP
1475       vars_of_eldap ();
1476 #endif
1477
1478       /* Now initialize any specifier variables.  We do this later
1479          because it has some dependence on the vars initialized
1480          above.
1481
1482          These functions should *only* initialize specifier variables,
1483          and may make use of the following functions/macros in addition
1484          to the ones listed above:
1485
1486          DEFVAR_SPECIFIER()
1487          Fmake_specifier()
1488          set_specifier_fallback()
1489          set_specifier_caching()
1490          */
1491
1492       specifier_vars_of_glyphs ();
1493 #ifdef HAVE_MENUBARS
1494       specifier_vars_of_menubar ();
1495 #endif
1496       specifier_vars_of_redisplay ();
1497 #ifdef HAVE_SCROLLBARS
1498       specifier_vars_of_scrollbar ();
1499 #endif
1500 #ifdef HAVE_TOOLBARS
1501       specifier_vars_of_toolbar ();
1502 #endif
1503       specifier_vars_of_window ();
1504
1505       /* Now comes all the rest of the variables that couldn't
1506          be handled above.  There may be dependencies on variables
1507          initialized above, and dependencies between one complex_vars_()
1508          function and another. */
1509
1510       /* Calls Fmake_range_table(). */
1511       complex_vars_of_regex ();
1512       /* Calls Fmake_range_table(). */
1513       complex_vars_of_search ();
1514
1515       /* Calls make_lisp_hash_table(). */
1516       complex_vars_of_extents ();
1517
1518       /* Depends on hash tables and specifiers. */
1519       complex_vars_of_faces ();
1520
1521 #ifdef MULE
1522       /* These two depend on hash tables and various variables declared
1523          earlier.  The second may also depend on the first. */
1524       complex_vars_of_mule_charset ();
1525 #endif
1526 #if defined(FILE_CODING)
1527       complex_vars_of_file_coding ();
1528 #endif
1529
1530       /* This calls allocate_glyph(), which creates specifiers
1531          and also relies on a variable (Vthe_nothing_vector) initialized
1532          above.  It also calls make_ext_string(), which under Mule
1533          could require that the charsets be initialized. */
1534       complex_vars_of_glyphs ();
1535
1536       /* These rely on the glyphs just created in the previous function,
1537          and call Fadd_spec_to_specifier(), which relies on various
1538          variables initialized above. */
1539 #ifdef HAVE_X_WINDOWS
1540       complex_vars_of_glyphs_x ();
1541 #endif
1542 #ifdef HAVE_MS_WINDOWS
1543       complex_vars_of_glyphs_mswindows ();
1544 #endif
1545
1546       /* This calls Fmake_glyph_internal(). */
1547       complex_vars_of_alloc ();
1548
1549       /* This calls Fmake_glyph_internal(). */
1550 #ifdef HAVE_MENUBARS
1551       complex_vars_of_menubar ();
1552 #endif
1553
1554       /* This calls Fmake_glyph_internal(). */
1555 #ifdef HAVE_SCROLLBARS
1556       complex_vars_of_scrollbar ();
1557 #endif
1558
1559       /* This calls allocate_glyph(). */
1560       complex_vars_of_frame ();
1561
1562       /* This calls Fcopy_category_table() under Mule, which calls who
1563          knows what. */
1564       complex_vars_of_chartab ();
1565
1566       /* This calls set_string_char(), which (under Mule) depends on the
1567          charsets being initialized. */
1568       complex_vars_of_casetab ();
1569
1570       /* This calls Fcopy_syntax_table(), which relies on char tables. */
1571       complex_vars_of_syntax ();
1572
1573       /* This initializes buffer-local variables, sets things up so
1574          that buffers can be created, and creates a couple of basic
1575          buffers.  This depends on Vstandard_syntax_table and
1576          Vstandard_category_table (initialized in the previous
1577          functions), as well as a whole horde of variables that may
1578          have been initialized above. */
1579       complex_vars_of_buffer ();
1580
1581       /* This initializes console-local variables. */
1582       complex_vars_of_console ();
1583
1584       /* This creates a couple more buffers, and depends on the
1585          previous function. */
1586       complex_vars_of_minibuf ();
1587
1588       /* These two might call Ffile_name_as_directory(), which
1589          might depend on all sorts of things; I'm not sure. */
1590       complex_vars_of_emacs ();
1591
1592       /* This creates a couple of basic keymaps and depends on Lisp
1593          hash tables and Ffset() (both of which depend on some variables
1594          initialized in the vars_of_*() section) and possibly other
1595          stuff. */
1596       complex_vars_of_keymap ();
1597
1598       /* Calls make_lisp_hash_table() and creates a keymap */
1599       complex_vars_of_event_stream ();
1600
1601 #ifdef ERROR_CHECK_GC
1602       {
1603         extern int always_gc;
1604         if (always_gc)                /* purification debugging hack */
1605           garbage_collect_1 ();
1606       }
1607 #endif
1608     }
1609
1610   /* CONGRATULATIONS!!!  We have successfully initialized the Lisp
1611      engine. */
1612
1613   if (initialized)
1614     {
1615       /* Stuff that needs to be reset at run time.  Order below should
1616          not matter. */
1617       reinit_alloc ();
1618       reinit_eval ();
1619 #ifdef MULE_REGEXP
1620       reinit_mule_category ();
1621 #endif
1622     }
1623
1624   /* Now do further initialization/setup of stuff that is not needed by the
1625      syms_of_() routines.  This involves stuff that only is enabled in
1626      an interactive run (redisplay, user input, etc.) and stuff that is
1627      not needed until we start loading Lisp code (the reader).  A lot
1628      of this stuff involves querying the current environment and needs
1629      to be done both at dump time and at run time. */
1630
1631   init_initial_directory();             /* get the directory to use for the
1632                                            "*scratch*" buffer, etc. */
1633
1634 #ifdef WINDOWSNT
1635   /*
1636    * For Win32, call init_environment() now, so that environment/registry
1637    * variables will be properly entered into Vprocess_environment.
1638    */
1639   init_environment();
1640 #endif
1641
1642   init_callproc ();     /* Set up the process environment (so that egetenv
1643                            works), the basic directory variables
1644                            (exec-directory and so on), and stuff
1645                            related to subprocesses.  This should be
1646                            first because many of the functions below
1647                            call egetenv() to get environment variables. */
1648   init_lread ();        /* Set up the Lisp reader. */
1649 #ifdef MSDOS
1650   /* Call early 'cause init_environment needs it.  */
1651   init_dosfns ();
1652   /* Set defaults for several environment variables.  */
1653   init_environment (argc, argv, skip_args);
1654 #endif
1655   init_cmdargs (argc, argv, skip_args); /* Create list Vcommand_line_args */
1656   init_buffer ();       /* Set default directory of *scratch* buffer */
1657
1658 #ifdef WINDOWSNT
1659   init_ntproc();
1660 #endif
1661
1662   init_redisplay ();      /* Determine terminal type.
1663                              init_sys_modes uses results */
1664   init_event_stream (); /* Set up so we can get user input. */
1665   init_macros (); /* set up so we can run macros. */
1666   init_editfns (); /* Determine the name of the user we're running as */
1667   init_xemacs_process (); /* set up for calling subprocesses */
1668 #ifdef SUNPRO
1669   init_sunpro (); /* Set up Sunpro usage tracking */
1670 #endif
1671 #if defined (HAVE_NATIVE_SOUND) && defined (hp9000s800)
1672   init_hpplay ();
1673 #endif
1674 #ifdef HAVE_TTY
1675   init_device_tty ();
1676 #endif
1677   init_console_stream (); /* Create the first console */
1678
1679   /* try to get the actual pathname of the exec file we are running */
1680   if (!restart)
1681   {
1682     Vinvocation_name = Fcar (Vcommand_line_args);
1683     if (XSTRING_DATA(Vinvocation_name)[0] == '-')
1684       {
1685         /* XEmacs as a login shell, oh goody! */
1686         Vinvocation_name = build_string(getenv("SHELL"));
1687       }
1688     Vinvocation_directory = Vinvocation_name;
1689
1690     if (!NILP (Ffile_name_directory (Vinvocation_name)))
1691       {
1692         /* invocation-name includes a directory component -- presumably it
1693            is relative to cwd, not $PATH */
1694         Vinvocation_directory = Fexpand_file_name (Vinvocation_name,
1695                                                    Qnil);
1696         Vinvocation_path = Qnil;
1697       }
1698     else
1699       {
1700         Vinvocation_path = decode_env_path ("PATH", NULL);
1701         locate_file (Vinvocation_path, Vinvocation_name,
1702                      Vlisp_EXEC_SUFFIXES,
1703                      &Vinvocation_directory, X_OK);
1704       }
1705
1706     if (NILP (Vinvocation_directory))
1707       Vinvocation_directory = Vinvocation_name;
1708
1709     Vinvocation_name = Ffile_name_nondirectory (Vinvocation_directory);
1710     Vinvocation_directory = Ffile_name_directory (Vinvocation_directory);
1711   }
1712
1713 #if defined(HAVE_SHLIB) && !defined(WINDOWSNT)
1714   /* This is Unix only.  MS Windows NT has a library call that does
1715      The Right Thing on that system.  Rumor has it, this must be
1716      called for GNU dld in temacs and xemacs.  */
1717   {
1718     char *buf = (char *)alloca (XSTRING_LENGTH (Vinvocation_directory)
1719                                 + XSTRING_LENGTH (Vinvocation_name)
1720                                 + 2);
1721     sprintf (buf, "%s/%s", XSTRING_DATA (Vinvocation_directory),
1722              XSTRING_DATA (Vinvocation_name));
1723
1724     /* All we can do is cry if an error happens, so ignore it. */
1725     (void) dll_init (buf);
1726   }
1727 #endif
1728
1729 #if defined (LOCALTIME_CACHE) && defined (HAVE_TZSET)
1730   /* sun's localtime() has a bug.  it caches the value of the time
1731      zone rather than looking it up every time.  Since localtime() is
1732      called to bolt the undumping time into the undumped emacs, this
1733      results in localtime() ignoring the TZ environment variable.
1734      This flushes the new TZ value into localtime(). */
1735   tzset ();
1736 #endif /* LOCALTIME_CACHE and TZSET */
1737
1738   load_me = Qnil;
1739   if (!initialized)
1740     {
1741       /* Handle -l loadup-and-dump, args passed by Makefile. */
1742       if (argc > 2 + skip_args && !strcmp (argv[1 + skip_args], "-l"))
1743         load_me = build_string (argv[2 + skip_args]);
1744 #if 0 /* CANNOT_DUMP - this can never be right in XEmacs --andyp */
1745       /* Unless next switch is -nl, load "loadup.el" first thing.  */
1746       if (!(argc > 1 + skip_args && !strcmp (argv[1 + skip_args], "-nl")))
1747         load_me = build_string ("loadup.el");
1748 #endif /* CANNOT_DUMP */
1749     }
1750
1751 #ifdef QUANTIFY
1752   if (initialized)
1753     quantify_start_recording_data ();
1754 #endif /* QUANTIFY */
1755
1756   initialized = 1;
1757
1758   /* This never returns.  */
1759   initial_command_loop (load_me);
1760   /* NOTREACHED */
1761 }
1762
1763 \f
1764 /* Sort the args so we can find the most important ones
1765    at the beginning of argv.  */
1766
1767 /* First, here's a table of all the standard options.  */
1768
1769 struct standard_args
1770 {
1771   CONST char * CONST name;
1772   CONST char * CONST longname;
1773   int priority;
1774   int nargs;
1775 };
1776
1777 static struct standard_args standard_args[] =
1778 {
1779   /* Handled by main_1 above: */
1780   { "-nl", "--no-shared-memory", 100, 0 },
1781   { "-t", "--terminal", 95, 1 },
1782   { "-nw", "--no-windows", 90, 0 },
1783   { "-batch", "--batch", 85, 0 },
1784   { "-debug-paths", "--debug-paths", 82, 0 },
1785   { "-help", "--help", 80, 0 },
1786   { "-version", "--version", 75, 0 },
1787   { "-V", 0, 75, 0 },
1788   { "-d", "--display", 80, 1 },
1789   { "-display", 0, 80, 1 },
1790   { "-NXHost",  0, 79, 0 },
1791   { "-MachLaunch", 0, 79, 0},
1792
1793   /* Handled by command-line-early in startup.el: */
1794   { "-q", "--no-init-file", 50, 0 },
1795   { "-unmapped", 0, 50, 0 },
1796   { "-no-init-file", 0, 50, 0 },
1797   { "-vanilla", "--vanilla", 50, 0 },
1798   { "-no-autoloads", "--no-autoloads", 50, 0 },
1799   { "-no-site-file", "--no-site-file", 40, 0 },
1800   { "-no-early-packages", "--no-early-packages", 35, 0 },
1801   { "-u", "--user", 30, 1 },
1802   { "-user", 0, 30, 1 },
1803   { "-debug-init", "--debug-init", 20, 0 },
1804   { "-debug-paths", "--debug-paths", 20, 0 },
1805
1806   /* Xt options: */
1807   { "-i", "--icon-type", 15, 0 },
1808   { "-itype", 0, 15, 0 },
1809   { "-iconic", "--iconic", 15, 0 },
1810   { "-bg", "--background-color", 10, 1 },
1811   { "-background", 0, 10, 1 },
1812   { "-fg", "--foreground-color", 10, 1 },
1813   { "-foreground", 0, 10, 1 },
1814   { "-bd", "--border-color", 10, 1 },
1815   { "-bw", "--border-width", 10, 1 },
1816   { "-ib", "--internal-border", 10, 1 },
1817   { "-ms", "--mouse-color", 10, 1 },
1818   { "-cr", "--cursor-color", 10, 1 },
1819   { "-fn", "--font", 10, 1 },
1820   { "-font", 0, 10, 1 },
1821   { "-g", "--geometry", 10, 1 },
1822   { "-geometry", 0, 10, 1 },
1823   { "-T", "--title", 10, 1 },
1824   { "-title", 0, 10, 1 },
1825   { "-name", "--name", 10, 1 },
1826   { "-xrm", "--xrm", 10, 1 },
1827   { "-r", "--reverse-video", 5, 0 },
1828   { "-rv", 0, 5, 0 },
1829   { "-reverse", 0, 5, 0 },
1830   { "-hb", "--horizontal-scroll-bars", 5, 0 },
1831   { "-vb", "--vertical-scroll-bars", 5, 0 },
1832
1833   /* These have the same priority as ordinary file name args,
1834      so they are not reordered with respect to those.  */
1835   { "-L", "--directory", 0, 1 },
1836   { "-directory", 0, 0, 1 },
1837   { "-l", "--load", 0, 1 },
1838   { "-load", 0, 0, 1 },
1839   { "-f", "--funcall", 0, 1 },
1840   { "-funcall", 0, 0, 1 },
1841   { "-eval", "--eval", 0, 1 },
1842   { "-insert", "--insert", 0, 1 },
1843   /* This should be processed after ordinary file name args and the like.  */
1844   { "-kill", "--kill", -10, 0 },
1845 };
1846
1847 /* Reorder the elements of ARGV (assumed to have ARGC elements)
1848    so that the highest priority ones come first.
1849    Do not change the order of elements of equal priority.
1850    If an option takes an argument, keep it and its argument together.  */
1851
1852 static void
1853 sort_args (int argc, char **argv)
1854 {
1855   char **new_argv = xnew_array (char *, argc);
1856   /* For each element of argv,
1857      the corresponding element of options is:
1858      0 for an option that takes no arguments,
1859      1 for an option that takes one argument, etc.
1860      -1 for an ordinary non-option argument.  */
1861   int *options  = xnew_array (int, argc);
1862   int *priority = xnew_array (int, argc);
1863   int to = 1;
1864   int from;
1865   int i;
1866   int end_of_options_p = 0;
1867
1868   /* Categorize all the options,
1869      and figure out which argv elts are option arguments.  */
1870   for (from = 1; from < argc; from++)
1871     {
1872       options[from] = -1;
1873       priority[from] = 0;
1874       /* Pseudo options "--" and "run-temacs" indicate end of options */
1875       if (!strcmp (argv[from], "--") ||
1876           !strcmp (argv[from], "run-temacs"))
1877         end_of_options_p = 1;
1878       if (!end_of_options_p && argv[from][0] == '-')
1879         {
1880           int match, thislen;
1881           char *equals;
1882
1883           /* Look for a match with a known old-fashioned option.  */
1884           for (i = 0; i < countof (standard_args); i++)
1885             if (!strcmp (argv[from], standard_args[i].name))
1886               {
1887                 options[from]  = standard_args[i].nargs;
1888                 priority[from] = standard_args[i].priority;
1889                 if (from + standard_args[i].nargs >= argc)
1890                   fatal ("Option `%s' requires an argument\n", argv[from]);
1891                 from += standard_args[i].nargs;
1892                 goto done;
1893               }
1894
1895           /* Look for a match with a known long option.
1896              MATCH is -1 if no match so far, -2 if two or more matches so far,
1897              >= 0 (the table index of the match) if just one match so far.  */
1898           if (argv[from][1] == '-')
1899             {
1900               match = -1;
1901               thislen = strlen (argv[from]);
1902               equals = strchr (argv[from], '=');
1903               if (equals != 0)
1904                 thislen = equals - argv[from];
1905
1906               for (i = 0; i < countof (standard_args); i++)
1907                 if (standard_args[i].longname
1908                     && !strncmp (argv[from], standard_args[i].longname,
1909                                  thislen))
1910                   {
1911                     if (match == -1)
1912                       match = i;
1913                     else
1914                       match = -2;
1915                   }
1916
1917               /* If we found exactly one match, use that.  */
1918               if (match >= 0)
1919                 {
1920                   options[from]  = standard_args[match].nargs;
1921                   priority[from] = standard_args[match].priority;
1922                   /* If --OPTION=VALUE syntax is used,
1923                      this option uses just one argv element.  */
1924                   if (equals != 0)
1925                     options[from] = 0;
1926                   if (from + options[from] >= argc)
1927                     fatal ("Option `%s' requires an argument\n", argv[from]);
1928                   from += options[from];
1929                 }
1930             }
1931         done: ;
1932         }
1933     }
1934
1935   /* Copy the arguments, in order of decreasing priority, to NEW_ARGV.  */
1936   new_argv[0] = argv[0];
1937   while (to < argc)
1938     {
1939       int best = -1;
1940       int best_priority = -9999;
1941
1942       /* Find the highest priority remaining option.
1943          If several have equal priority, take the first of them.  */
1944       for (from = 1; from < argc; from++)
1945         {
1946           if (argv[from] != 0 && priority[from] > best_priority)
1947             {
1948               best_priority = priority[from];
1949               best = from;
1950             }
1951           /* Skip option arguments--they are tied to the options.  */
1952           if (options[from] > 0)
1953             from += options[from];
1954         }
1955
1956       if (best < 0)
1957         abort ();
1958
1959       /* Copy the highest priority remaining option, with its args, to NEW_ARGV.  */
1960       new_argv[to++] = argv[best];
1961       for (i = 0; i < options[best]; i++)
1962         new_argv[to++] = argv[best + i + 1];
1963
1964       /* Clear out this option in ARGV.  */
1965       argv[best] = 0;
1966       for (i = 0; i < options[best]; i++)
1967         argv[best + i + 1] = 0;
1968     }
1969
1970   memcpy (argv, new_argv, sizeof (char *) * argc);
1971   xfree (new_argv);
1972   xfree (options);
1973   xfree (priority);
1974 }
1975
1976 static JMP_BUF run_temacs_catch;
1977
1978 static int run_temacs_argc;
1979 static char **run_temacs_argv;
1980 static char *run_temacs_args;
1981 static size_t run_temacs_argv_size;
1982 static size_t run_temacs_args_size;
1983
1984 DEFUN ("running-temacs-p", Frunning_temacs_p, 0, 0, 0, /*
1985 True if running temacs.  This means we are in the dumping stage.
1986 This is false during normal execution of the `xemacs' program, and
1987 becomes false once `run-emacs-from-temacs' is run.
1988 */
1989        ())
1990 {
1991   return run_temacs_argc >= 0 ? Qt : Qnil;
1992 }
1993
1994 DEFUN ("run-emacs-from-temacs", Frun_emacs_from_temacs, 0, MANY, 0, /*
1995 Do not call this.  It will reinitialize your XEmacs.  You'll be sorry.
1996 */
1997 /* If this function is called from startup.el, it will be possible to run
1998    temacs as an editor using 'temacs -batch -l loadup.el run-temacs', instead
1999    of having to dump an emacs and then run that (when debugging emacs itself,
2000    this can be much faster)). [Actually, the speed difference isn't that
2001    much as long as your filesystem is local, and you don't end up with
2002    a dumped version in case you want to rerun it.  This function is most
2003    useful when used as part of the `make all-elc' command. --ben]
2004    This will "restart" emacs with the specified command-line arguments.
2005
2006    Martin thinks this function is most useful when using debugging
2007    tools like Purify or tcov that get confused by XEmacs' dumping.  */
2008      (int nargs, Lisp_Object *args))
2009 {
2010   int ac;
2011   CONST Extbyte *wampum;
2012   int namesize;
2013   int total_len;
2014   Lisp_Object orig_invoc_name = Fcar (Vcommand_line_args);
2015   CONST Extbyte **wampum_all = alloca_array (CONST Extbyte *, nargs);
2016   int *wampum_all_len  = alloca_array (int, nargs);
2017
2018   assert (!gc_in_progress);
2019
2020   if (run_temacs_argc < 0)
2021     error ("I've lost my temacs-hood.");
2022
2023   /* Need to convert the orig_invoc_name and all of the arguments
2024      to external format. */
2025
2026   GET_STRING_EXT_DATA_ALLOCA (orig_invoc_name, FORMAT_OS, wampum,
2027                               namesize);
2028   namesize++;
2029
2030   for (ac = 0, total_len = namesize; ac < nargs; ac++)
2031     {
2032       CHECK_STRING (args[ac]);
2033       GET_STRING_EXT_DATA_ALLOCA (args[ac], FORMAT_OS,
2034                                   wampum_all[ac],
2035                                   wampum_all_len[ac]);
2036       wampum_all_len[ac]++;
2037       total_len += wampum_all_len[ac];
2038     }
2039   DO_REALLOC (run_temacs_args, run_temacs_args_size, total_len, char);
2040   DO_REALLOC (run_temacs_argv, run_temacs_argv_size, nargs+2, char *);
2041
2042   memcpy (run_temacs_args, wampum, namesize);
2043   run_temacs_argv [0] = run_temacs_args;
2044   for (ac = 0; ac < nargs; ac++)
2045     {
2046       memcpy (run_temacs_args + namesize,
2047               wampum_all[ac], wampum_all_len[ac]);
2048       run_temacs_argv [ac + 1] = run_temacs_args + namesize;
2049       namesize += wampum_all_len[ac];
2050     }
2051   run_temacs_argv [nargs + 1] = 0;
2052   catchlist = NULL; /* Important!  Otherwise free_cons() calls in
2053                        condition_case_unwind() may lead to GC death. */
2054   unbind_to (0, Qnil); /* this closes loadup.el */
2055   purify_flag = 0;
2056   run_temacs_argc = nargs + 1;
2057 #ifdef HEAP_IN_DATA
2058   report_sheap_usage (0);
2059 #endif
2060   LONGJMP (run_temacs_catch, 1);
2061   return Qnil; /* not reached; warning suppression */
2062 }
2063
2064 /* ARGSUSED */
2065 int
2066 main (int argc, char **argv, char **envp)
2067 {
2068   int     volatile vol_argc = argc;
2069   char ** volatile vol_argv = argv;
2070   char ** volatile vol_envp = envp;
2071   /* This is hairy.  We need to compute where the XEmacs binary was invoked
2072      from because temacs initialization requires it to find the lisp
2073      directories.  The code that recomputes the path is guarded by the
2074      restarted flag.  There are three possible paths I've found so far
2075      through this:
2076
2077      temacs -- When running temacs for basic build stuff, the first main_1
2078       will be the only one invoked.  It must compute the path else there
2079       will be a very ugly bomb in startup.el (can't find obvious location
2080       for doc-directory data-directory, etc.).
2081
2082      temacs w/ run-temacs on the command line -- This is run to bytecompile
2083       all the out of date dumped lisp.  It will execute both of the main_1
2084       calls and the second one must not touch the first computation because
2085       argc/argv are hosed the second time through.
2086
2087      xemacs -- Only the second main_1 is executed.  The invocation path must
2088       computed but this only matters when running in place or when running
2089       as a login shell.
2090
2091      As a bonus for straightening this out, XEmacs can now be run in place
2092      as a login shell.  This never used to work.
2093
2094      As another bonus, we can now guarantee that
2095      (concat invocation-directory invocation-name) contains the filename
2096      of the XEmacs binary we are running.  This can now be used in a
2097      definite test for out of date dumped files.  -slb */
2098   int restarted = 0;
2099 #ifdef QUANTIFY
2100   quantify_stop_recording_data ();
2101   quantify_clear_data ();
2102 #endif /* QUANTIFY */
2103
2104   suppress_early_error_handler_backtrace = 0;
2105   lim_data = 0; /* force reinitialization of this variable */
2106
2107   /* Lisp_Object must fit in a word; check VALBITS and GCTYPEBITS */
2108   assert (sizeof (Lisp_Object) == sizeof (void *));
2109
2110 #ifdef LINUX_SBRK_BUG
2111   sbrk (1);
2112 #endif
2113
2114   if (!initialized)
2115     {
2116 #ifdef DOUG_LEA_MALLOC
2117       mallopt (M_MMAP_MAX, 0);
2118 #endif
2119       run_temacs_argc = 0;
2120       if (! SETJMP (run_temacs_catch))
2121         {
2122           main_1 (vol_argc, vol_argv, vol_envp, 0);
2123         }
2124       /* run-emacs-from-temacs called */
2125       restarted = 1;
2126       vol_argc = run_temacs_argc;
2127       vol_argv = run_temacs_argv;
2128 #ifdef _SCO_DS
2129       /* This makes absolutely no sense to anyone involved.  There are
2130          several people using this stuff.  We've compared versions on
2131          everything we can think of.  We can find no difference.
2132          However, on both my systems environ is a plain old global
2133          variable initialized to zero.  _environ is the one that
2134          contains pointers to the actual environment.
2135
2136          Since we can't figure out the difference (and we're hours
2137          away from a release), this takes a very cowardly approach and
2138          is bracketed with both a system specific preprocessor test
2139          and a runtime "do you have this problem" test
2140
2141          06/20/96 robertl@dgii.com */
2142       {
2143         extern char *_environ;
2144         if ((unsigned) environ == 0)
2145           environ=_environ;
2146       }
2147 #endif /* _SCO_DS */
2148       vol_envp = environ;
2149     }
2150 #ifdef RUN_TIME_REMAP
2151   else
2152     /* obviously no-one uses this because where it was before initialized was
2153      *always* true */
2154     run_time_remap (argv[0]);
2155 #endif
2156
2157 #ifdef DOUG_LEA_MALLOC
2158   if (initialized && (malloc_state_ptr != NULL))
2159     {
2160       int rc = malloc_set_state (malloc_state_ptr);
2161       if (rc != 0)
2162         {
2163           fprintf (stderr, "malloc_set_state failed, rc = %d\n", rc);
2164           abort ();
2165         }
2166 #if 0
2167       free (malloc_state_ptr);
2168 #endif
2169       /* mmap works in glibc-2.1, glibc-2.0 (Non-Mule only) and Linux libc5 */
2170 #if (defined(__GLIBC__) && __GLIBC_MINOR__ >= 1) || \
2171     defined(_NO_MALLOC_WARNING_) || \
2172     (defined(__GLIBC__) && __GLIBC_MINOR__ < 1 && !defined(MULE)) || \
2173     defined(DEBUG_DOUG_LEA_MALLOC)
2174       mallopt (M_MMAP_MAX, 64);
2175 #endif
2176 #ifdef REL_ALLOC
2177       r_alloc_reinit ();
2178 #endif
2179     }
2180 #endif /* DOUG_LEA_MALLOC */
2181
2182   run_temacs_argc = -1;
2183
2184   main_1 (vol_argc, vol_argv, vol_envp, restarted);
2185   return 0; /* unreached */
2186 }
2187
2188 \f
2189 /* Dumping apparently isn't supported by versions of GCC >= 2.8. */
2190 /* The following needs conditionalization on whether either XEmacs or */
2191 /* various system shared libraries have been built and linked with */
2192 /* GCC >= 2.8.  -slb */
2193 #if defined(GNU_MALLOC)
2194 static void
2195 voodoo_free_hook (void *mem)
2196 {
2197   /* Disable all calls to free() when XEmacs is exiting and it doesn't */
2198   /* matter. */
2199   __free_hook = voodoo_free_hook;
2200 }
2201 #endif
2202
2203 DEFUN ("kill-emacs", Fkill_emacs, 0, 1, "P", /*
2204 Exit the XEmacs job and kill it.  Ask for confirmation, without argument.
2205 If ARG is an integer, return ARG as the exit program code.
2206 If ARG is a string, stuff it as keyboard input.
2207
2208 The value of `kill-emacs-hook', if not void,
2209 is a list of functions (of no args),
2210 all of which are called before XEmacs is actually killed.
2211 */
2212        (arg))
2213 {
2214   /* This function can GC */
2215   struct gcpro gcpro1;
2216
2217   GCPRO1 (arg);
2218
2219   if (feof (stdin))
2220     arg = Qt;
2221
2222   if (!preparing_for_armageddon && !noninteractive)
2223     run_hook (Qkill_emacs_hook);
2224
2225   /* make sure no quitting from now on!! */
2226   dont_check_for_quit = 1;
2227   Vinhibit_quit = Qt;
2228
2229   if (!preparing_for_armageddon)
2230     {
2231       Lisp_Object concons, nextcons;
2232
2233       /* Normally, go ahead and delete all the consoles now.
2234          Some unmentionably lame window systems (MS Wwwww...... eek,
2235          I can't even say it) don't properly clean up after themselves,
2236          and even for those that do, it might be cleaner this way.
2237          If we're going down, however, we don't do this (might
2238          be too dangerous), and if we get a crash somewhere within
2239          this loop, we'll still autosave and won't try this again. */
2240
2241       LIST_LOOP_DELETING(concons, nextcons, Vconsole_list)
2242         {
2243           /* There is very little point in deleting the stream console.
2244              It uses stdio, which should flush any buffered output and
2245              something can only go wrong. -slb */
2246           /* I changed my mind.  There's a stupid hack in close to add
2247              a trailing newline. */
2248           /*if (!CONSOLE_STREAM_P (XCONSOLE (XCAR (concons))))*/
2249             delete_console_internal (XCONSOLE (XCAR (concons)), 1, 1, 0);
2250         }
2251     }
2252
2253   UNGCPRO;
2254
2255   shut_down_emacs (0, ((STRINGP (arg)) ? arg : Qnil));
2256
2257 #if defined(GNU_MALLOC)
2258   __free_hook = voodoo_free_hook;
2259 #endif
2260
2261   exit ((INTP (arg)) ? XINT (arg) : 0);
2262   /* NOTREACHED */
2263   return Qnil; /* I'm sick of the compiler warning */
2264 }
2265
2266 /* Perform an orderly shutdown of XEmacs.  Autosave any modified
2267    buffers, kill any child processes, clean up the terminal modes (if
2268    we're in the foreground), and other stuff like that.  Don't perform
2269    any redisplay; this may be called when XEmacs is shutting down in
2270    the background, or after its X connection has died.
2271
2272    If SIG is a signal number, print a message for it.
2273
2274    This is called by fatal signal handlers, X protocol error handlers,
2275    and Fkill_emacs.  */
2276 static void
2277 shut_down_emacs (int sig, Lisp_Object stuff)
2278 {
2279   /* This function can GC */
2280   /* Prevent running of hooks and other non-essential stuff
2281      from now on.  */
2282   preparing_for_armageddon = 1;
2283
2284   /* In case frames or windows are screwed up, avoid assertion
2285      failures here */
2286   Vinhibit_quit = Qt;
2287
2288 #ifdef QUANTIFY
2289   quantify_stop_recording_data ();
2290 #endif /* QUANTIFY */
2291
2292 #if 0
2293   /* This is absolutely the most important thing to do, so make sure
2294      we do it now, before anything else.  We might have crashed and
2295      be in a weird inconsistent state, and potentially anything could
2296      set off another protection fault and cause us to bail out
2297      immediately. */
2298   /* I'm not removing the code entirely, yet.  We have run up against
2299      a spate of problems in diagnosing crashes due to crashes within
2300      crashes.  It has very definitely been determined that code called
2301      during auto-saving cannot work if XEmacs crashed inside of GC.
2302      We already auto-save on an itimer so there cannot be too much
2303      unsaved stuff around, and if we get better crash reports we might
2304      be able to get more problems fixed so I'm disabling this.  -slb */
2305   Fdo_auto_save (Qt, Qnil); /* do this before anything hazardous */
2306 #endif
2307
2308   fflush (stdout);
2309   reset_all_consoles ();
2310   if (sig && sig != SIGTERM)
2311     {
2312       stderr_out ("\nFatal error (%d).\n", sig);
2313       stderr_out
2314         ("Your files have been auto-saved.\n"
2315          "Use `M-x recover-session' to recover them.\n"
2316          "\n"
2317 #ifdef INFODOCK
2318          "Please report this bug by selecting `Report-Bug' in the InfoDock\n"
2319          "menu.\n"
2320 #else
2321          "Please report this bug by running the send-pr script included\n"
2322          "with XEmacs, or selecting `Send Bug Report' from the help menu.\n"
2323          "As a last resort send ordinary email to `crashes@xemacs.org'.\n"
2324 #endif
2325          "*MAKE SURE* to include the information in the command\n"
2326          "M-x describe-installation.\n"
2327          "\n"
2328          "If at all possible, *please* try to obtain a C stack backtrace;\n"
2329          "it will help us immensely in determining what went wrong.\n"
2330          "To do this, locate the core file that was produced as a result\n"
2331          "of this crash (it's usually called `core' and is located in the\n"
2332          "directory in which you started the editor, or maybe in your home\n"
2333          "directory), and type\n"
2334          "\n"
2335          "  gdb ");
2336       {
2337         CONST char *name;
2338         char *dir = 0;
2339
2340         /* Now try to determine the actual path to the executable,
2341            to try to make the backtrace-determination process as foolproof
2342            as possible. */
2343         if (GC_STRINGP (Vinvocation_name))
2344           name = (char *) XSTRING_DATA (Vinvocation_name);
2345         else
2346           name = "xemacs";
2347         if (GC_STRINGP (Vinvocation_directory))
2348           dir = (char *) XSTRING_DATA (Vinvocation_directory);
2349         if (!dir || dir[0] != '/')
2350           stderr_out ("`which %s`", name);
2351         else if (dir[strlen (dir) - 1] != '/')
2352           stderr_out ("%s/%s", dir, name);
2353         else
2354           stderr_out ("%s%s", dir, name);
2355       }
2356       stderr_out
2357         (" core\n\n"
2358          "then type `where' when the debugger prompt comes up.\n"
2359          "(If you don't have GDB on your system, you might have DBX,\n"
2360          "or XDB, or SDB.  A similar procedure should work for all of\n"
2361          "these.  Ask your system administrator if you need more help.)\n");
2362     }
2363
2364   stuff_buffered_input (stuff);
2365
2366   kill_buffer_processes (Qnil);
2367
2368 #ifdef CLASH_DETECTION
2369   unlock_all_files ();
2370 #endif
2371
2372 #ifdef TOOLTALK
2373   tt_session_quit (tt_default_session ());
2374 #if 0
2375   /* The following crashes when built on X11R5 and run on X11R6 */
2376   tt_close ();
2377 #endif
2378 #endif /* TOOLTALK */
2379
2380 }
2381
2382 \f
2383 #ifndef CANNOT_DUMP
2384 /* Nothing like this can be implemented on an Apollo.
2385    What a loss!  */
2386
2387 extern char my_edata[];
2388
2389 #ifdef HAVE_SHM
2390
2391 DEFUN ("dump-emacs-data", Fdump_emacs_data, 1, 1, 0, /*
2392 Dump current state of XEmacs into data file FILENAME.
2393 This function exists on systems that use HAVE_SHM.
2394 */
2395        (intoname))
2396 {
2397   /* This function can GC */
2398   int opurify;
2399   struct gcpro gcpro1;
2400   GCPRO1 (intoname);
2401
2402   CHECK_STRING (intoname);
2403   intoname = Fexpand_file_name (intoname, Qnil);
2404
2405   opurify = purify_flag;
2406   purify_flag = 0;
2407
2408   fflush (stderr);
2409   fflush (stdout);
2410
2411   disksave_object_finalization ();
2412   release_breathing_space ();
2413
2414   /* Tell malloc where start of impure now is */
2415   /* Also arrange for warnings when nearly out of space.  */
2416 #ifndef SYSTEM_MALLOC
2417   memory_warnings (my_edata, malloc_warning);
2418 #endif
2419   UNGCPRO;
2420   map_out_data (XSTRING_DATA (intoname));
2421
2422   purify_flag = opurify;
2423
2424   return Qnil;
2425 }
2426
2427 #else /* not HAVE_SHM */
2428 extern void disable_free_hook (void);
2429
2430 DEFUN ("dump-emacs", Fdump_emacs, 2, 2, 0, /*
2431 Dump current state of XEmacs into executable file FILENAME.
2432 Take symbols from SYMFILE (presumably the file you executed to run XEmacs).
2433 This is used in the file `loadup.el' when building XEmacs.
2434
2435 Remember to set `command-line-processed' to nil before dumping
2436 if you want the dumped XEmacs to process its command line
2437 and announce itself normally when it is run.
2438 */
2439        (intoname, symname))
2440 {
2441   /* This function can GC */
2442   struct gcpro gcpro1, gcpro2;
2443   int opurify;
2444
2445   GCPRO2 (intoname, symname);
2446
2447 #ifdef FREE_CHECKING
2448   Freally_free (Qnil);
2449
2450   /* When we're dumping, we can't use the debugging free() */
2451   disable_free_hook ();
2452 #endif
2453
2454   CHECK_STRING (intoname);
2455   intoname = Fexpand_file_name (intoname, Qnil);
2456   if (!NILP (symname))
2457     {
2458       CHECK_STRING (symname);
2459       if (XSTRING_LENGTH (symname) > 0)
2460         symname = Fexpand_file_name (symname, Qnil);
2461       else
2462         symname = Qnil;
2463     }
2464
2465   opurify = purify_flag;
2466   purify_flag = 0;
2467
2468 #ifdef HEAP_IN_DATA
2469   report_sheap_usage (1);
2470 #endif
2471
2472   clear_message ();
2473
2474   fflush (stderr);
2475   fflush (stdout);
2476
2477   disksave_object_finalization ();
2478   release_breathing_space ();
2479
2480   /* Tell malloc where start of impure now is */
2481   /* Also arrange for warnings when nearly out of space.  */
2482 #ifndef SYSTEM_MALLOC
2483   memory_warnings (my_edata, malloc_warning);
2484 #endif
2485
2486   UNGCPRO;
2487
2488 #if defined (MSDOS) && defined (EMX)
2489   {
2490     int fd = open ((char *) XSTRING_DATA (intoname),
2491                    O_WRONLY|O_CREAT|O_TRUNC, S_IREAD|S_IWRITE);
2492     if (!fd) {
2493       error ("Failure operating on %s", XSTRING_DATA (intoname));
2494     } else {
2495       _core (fd);
2496       close (fd);
2497     }
2498   }
2499 #else /* not MSDOS and EMX */
2500   {
2501     char *intoname_ext;
2502     char *symname_ext;
2503
2504     GET_C_STRING_FILENAME_DATA_ALLOCA (intoname, intoname_ext);
2505     if (STRINGP (symname))
2506       GET_C_STRING_FILENAME_DATA_ALLOCA (symname, symname_ext);
2507     else
2508       symname_ext = 0;
2509
2510     garbage_collect_1 ();
2511 #ifdef DOUG_LEA_MALLOC
2512     malloc_state_ptr = malloc_get_state ();
2513 #endif
2514   /* here we break our rule that the filename conversion should
2515      be performed at the actual time that the system call is made.
2516      It's a whole lot easier to do the conversion here than to
2517      modify all the unexec routines to ensure that filename
2518      conversion is applied everywhere.  Don't worry about memory
2519      leakage because this call only happens once. */
2520     unexec (intoname_ext, symname_ext, (uintptr_t) my_edata, 0, 0);
2521 #ifdef DOUG_LEA_MALLOC
2522     free (malloc_state_ptr);
2523 #endif
2524   }
2525 #endif /* not MSDOS and EMX */
2526
2527   purify_flag = opurify;
2528
2529   return Qnil;
2530 }
2531
2532 #endif /* not HAVE_SHM */
2533
2534 #endif /* not CANNOT_DUMP */
2535 \f
2536 #ifndef SEPCHAR
2537 #define SEPCHAR ':'
2538 #endif
2539
2540 /* Split STRING into a list of substrings.  The substrings are the
2541    parts of original STRING separated by SEPCHAR.  */
2542 static Lisp_Object
2543 split_string_by_emchar_1 (CONST Bufbyte *string, Bytecount size,
2544                           Emchar sepchar)
2545 {
2546   Lisp_Object result = Qnil;
2547   CONST Bufbyte *end = string + size;
2548
2549   while (1)
2550     {
2551       CONST Bufbyte *p = string;
2552       while (p < end)
2553         {
2554           if (charptr_emchar (p) == sepchar)
2555             break;
2556           INC_CHARPTR (p);
2557         }
2558       result = Fcons (make_string (string, p - string), result);
2559       if (p < end)
2560         {
2561           string = p;
2562           INC_CHARPTR (string); /* skip sepchar */
2563         }
2564       else
2565         break;
2566     }
2567   return Fnreverse (result);
2568 }
2569
2570 /* The same as the above, except PATH is an external C string (it is
2571    converted as FORMAT_FILENAME), and sepchar is hardcoded to SEPCHAR
2572    (':' or whatever).  */
2573 Lisp_Object
2574 decode_path (CONST char *path)
2575 {
2576   int len;
2577   Bufbyte *newpath;
2578   if (!path)
2579     return Qnil;
2580
2581   GET_C_CHARPTR_INT_FILENAME_DATA_ALLOCA (path, newpath);
2582
2583   len = strlen ((const char *) newpath);
2584   /* #### Does this make sense?  It certainly does for
2585      decode_env_path(), but it looks dubious here.  Does any code
2586      depend on decode_path("") returning nil instead of an empty
2587      string?  */
2588   if (!len)
2589     return Qnil;
2590
2591   return split_string_by_emchar_1 (newpath, (Bytecount)len, SEPCHAR);
2592 }
2593
2594 Lisp_Object
2595 decode_env_path (CONST char *evarname, CONST char *default_)
2596 {
2597   CONST char *path = 0;
2598   if (evarname)
2599     path = egetenv (evarname);
2600   if (!path)
2601     path = default_;
2602   return decode_path (path);
2603 }
2604
2605 /* Ben thinks this function should not exist or be exported to Lisp.
2606    We use it to define split-path-string in subr.el (not!).  */
2607
2608 DEFUN ("split-string-by-char", Fsplit_string_by_char, 1, 2, 0, /*
2609 Split STRING into a list of substrings originally separated by SEPCHAR.
2610 */
2611        (string, sepchar))
2612 {
2613   CHECK_STRING (string);
2614   CHECK_CHAR (sepchar);
2615   return split_string_by_emchar_1 (XSTRING_DATA (string),
2616                                    XSTRING_LENGTH (string),
2617                                    XCHAR (sepchar));
2618 }
2619
2620 /* #### This was supposed to be in subr.el, but is used VERY early in
2621    the bootstrap process, so it goes here.  Damn.  */
2622
2623 DEFUN ("split-path", Fsplit_path, 1, 1, 0, /*
2624 Explode a search path into a list of strings.
2625 The path components are separated with the characters specified
2626 with `path-separator'.
2627 */
2628        (path))
2629 {
2630   CHECK_STRING (path);
2631
2632   while (!STRINGP (Vpath_separator)
2633          || (XSTRING_CHAR_LENGTH (Vpath_separator) != 1))
2634     Vpath_separator = signal_simple_continuable_error
2635       ("`path-separator' should be set to a single-character string",
2636        Vpath_separator);
2637
2638   return (split_string_by_emchar_1
2639           (XSTRING_DATA (path), XSTRING_LENGTH (path),
2640            charptr_emchar (XSTRING_DATA (Vpath_separator))));
2641 }
2642 \f
2643 DEFUN ("noninteractive", Fnoninteractive, 0, 0, 0, /*
2644 Non-nil return value means XEmacs is running without interactive terminal.
2645 */
2646        ())
2647 {
2648   return noninteractive ? Qt : Qnil;
2649 }
2650
2651 /* This flag is useful to define if you're under a debugger; this way, you
2652    can put a breakpoint of assert_failed() and debug multiple problems
2653    in one session without having to recompile. */
2654 /* #define ASSERTIONS_DONT_ABORT */
2655
2656 #ifdef USE_ASSERTIONS
2657 /* This highly dubious kludge ... shut up Jamie, I'm tired of your slagging. */
2658
2659 DOESNT_RETURN
2660 assert_failed (CONST char *file, int line, CONST char *expr)
2661 {
2662   stderr_out ("Fatal error: assertion failed, file %s, line %d, %s\n",
2663               file, line, expr);
2664 #undef abort    /* avoid infinite #define loop... */
2665 #if defined (WINDOWSNT) && defined (DEBUG_XEMACS)
2666   DebugBreak ();
2667 #elif !defined (ASSERTIONS_DONT_ABORT)
2668   abort ();
2669 #endif
2670 }
2671 #endif /* USE_ASSERTIONS */
2672
2673 #ifdef QUANTIFY
2674 DEFUN ("quantify-start-recording-data", Fquantify_start_recording_data,
2675        0, 0, "", /*
2676 Start recording Quantify data.
2677 */
2678        ())
2679 {
2680   quantify_start_recording_data ();
2681   return Qnil;
2682 }
2683
2684 DEFUN ("quantify-stop-recording-data", Fquantify_stop_recording_data,
2685        0, 0, "", /*
2686 Stop recording Quantify data.
2687 */
2688        ())
2689 {
2690   quantify_stop_recording_data ();
2691   return Qnil;
2692 }
2693
2694 DEFUN ("quantify-clear-data", Fquantify_clear_data, 0, 0, "", /*
2695 Clear all Quantify data.
2696 */
2697        ())
2698 {
2699   quantify_clear_data ();
2700   return Qnil;
2701 }
2702 #endif /* QUANTIFY */
2703
2704 void
2705 syms_of_emacs (void)
2706 {
2707 #ifndef CANNOT_DUMP
2708 #ifdef HAVE_SHM
2709   DEFSUBR (Fdump_emacs_data);
2710 #else
2711   DEFSUBR (Fdump_emacs);
2712 #endif
2713 #endif /* !CANNOT_DUMP */
2714
2715   DEFSUBR (Frun_emacs_from_temacs);
2716   DEFSUBR (Frunning_temacs_p);
2717   DEFSUBR (Finvocation_name);
2718   DEFSUBR (Finvocation_directory);
2719   DEFSUBR (Fkill_emacs);
2720   DEFSUBR (Fnoninteractive);
2721
2722 #ifdef QUANTIFY
2723   DEFSUBR (Fquantify_start_recording_data);
2724   DEFSUBR (Fquantify_stop_recording_data);
2725   DEFSUBR (Fquantify_clear_data);
2726 #endif /* QUANTIFY */
2727
2728   DEFSUBR (Fsplit_string_by_char);
2729   DEFSUBR (Fsplit_path);        /* #### */
2730
2731   defsymbol (&Qkill_emacs_hook, "kill-emacs-hook");
2732   defsymbol (&Qsave_buffers_kill_emacs, "save-buffers-kill-emacs");
2733 }
2734
2735 void
2736 vars_of_emacs (void)
2737 {
2738   DEFVAR_BOOL ("suppress-early-error-handler-backtrace",
2739                &suppress_early_error_handler_backtrace /*
2740 Non-nil means early error handler shouldn't print a backtrace.
2741 */ );
2742
2743   DEFVAR_LISP ("command-line-args", &Vcommand_line_args /*
2744 Args passed by shell to XEmacs, as a list of strings.
2745 */ );
2746
2747   DEFVAR_LISP ("invocation-name", &Vinvocation_name /*
2748 The program name that was used to run XEmacs.
2749 Any directory names are omitted.
2750 */ );
2751
2752   DEFVAR_LISP ("invocation-directory", &Vinvocation_directory /*
2753 The directory in which the XEmacs executable was found, to run it.
2754 The value is simply the program name if that directory's name is not known.
2755 */ );
2756
2757   DEFVAR_LISP ("invocation-path", &Vinvocation_path /*
2758 The path in which the XEmacs executable was found, to run it.
2759 The value is simply the value of environment variable PATH on startup
2760 if XEmacs was found there.
2761 */ );
2762
2763 #if 0 /* FSFmacs */
2764   xxDEFVAR_LISP ("installation-directory", &Vinstallation_directory,
2765     "A directory within which to look for the `lib-src' and `etc' directories.\n"
2766 "This is non-nil when we can't find those directories in their standard\n"
2767 "installed locations, but we can find them\n"
2768 "near where the XEmacs executable was found.");
2769 #endif
2770
2771   DEFVAR_LISP ("system-type", &Vsystem_type /*
2772 Symbol indicating type of operating system you are using.
2773 */ );
2774   Vsystem_type = intern (SYSTEM_TYPE);
2775   Fprovide (intern(SYSTEM_TYPE));
2776
2777 #ifndef EMACS_CONFIGURATION
2778 # define EMACS_CONFIGURATION "UNKNOWN"
2779 #endif
2780   DEFVAR_LISP ("system-configuration", &Vsystem_configuration /*
2781 String naming the configuration XEmacs was built for.
2782 */ );
2783   Vsystem_configuration = Fpurecopy (build_string (EMACS_CONFIGURATION));
2784
2785 #ifndef EMACS_CONFIG_OPTIONS
2786 # define EMACS_CONFIG_OPTIONS "UNKNOWN"
2787 #endif
2788   DEFVAR_LISP ("system-configuration-options", &Vsystem_configuration_options /*
2789 String containing the configuration options XEmacs was built with.
2790 */ );
2791   Vsystem_configuration_options = Fpurecopy (build_string
2792                                              (EMACS_CONFIG_OPTIONS));
2793
2794   DEFVAR_LISP ("emacs-major-version", &Vemacs_major_version /*
2795 Major version number of this version of Emacs, as an integer.
2796 Warning: this variable did not exist in Emacs versions earlier than:
2797   FSF Emacs:   19.23
2798   XEmacs:      19.10
2799 */ );
2800   Vemacs_major_version = make_int (EMACS_MAJOR_VERSION);
2801
2802   DEFVAR_LISP ("emacs-minor-version", &Vemacs_minor_version /*
2803 Minor version number of this version of Emacs, as an integer.
2804 Warning: this variable did not exist in Emacs versions earlier than:
2805   FSF Emacs:   19.23
2806   XEmacs:      19.10
2807 */ );
2808   Vemacs_minor_version = make_int (EMACS_MINOR_VERSION);
2809
2810   DEFVAR_LISP ("emacs-patch-level", &Vemacs_patch_level /*
2811 The patch level of this version of Emacs, as an integer.
2812 The value is non-nil if this version of XEmacs is part of a series of
2813 stable XEmacsen, but has bug fixes applied.
2814 Warning: this variable does not exist in FSF Emacs or in XEmacs versions
2815 earlier than 21.1.1
2816 */ );
2817 #ifdef EMACS_PATCH_LEVEL
2818   Vemacs_patch_level = make_int (EMACS_PATCH_LEVEL);
2819 #else
2820   Vemacs_patch_level = Qnil;
2821 #endif
2822
2823     DEFVAR_LISP ("emacs-beta-version", &Vemacs_beta_version /*
2824 Beta number of this version of Emacs, as an integer.
2825 The value is nil if this is an officially released version of XEmacs.
2826 Warning: this variable does not exist in FSF Emacs or in XEmacs versions
2827 earlier than 20.3.
2828 */ );
2829 #ifdef EMACS_BETA_VERSION
2830   Vemacs_beta_version = make_int (EMACS_BETA_VERSION);
2831 #else
2832   Vemacs_beta_version = Qnil;
2833 #endif
2834
2835 #ifdef INFODOCK
2836   DEFVAR_LISP ("infodock-major-version", &Vinfodock_major_version /*
2837 Major version number of this InfoDock release.
2838 */ );
2839   Vinfodock_major_version = make_int (INFODOCK_MAJOR_VERSION);
2840
2841   DEFVAR_LISP ("infodock-minor-version", &Vinfodock_minor_version /*
2842 Minor version number of this InfoDock release.
2843 */ );
2844   Vinfodock_minor_version = make_int (INFODOCK_MINOR_VERSION);
2845
2846   DEFVAR_LISP ("infodock-build-version", &Vinfodock_build_version /*
2847 Build version of this InfoDock release.
2848 */ );
2849   Vinfodock_build_version = make_int (INFODOCK_BUILD_VERSION);
2850 #endif
2851
2852   DEFVAR_LISP ("xemacs-codename", &Vxemacs_codename /*
2853 Codename of this version of Emacs (a string).
2854 */ );
2855 #ifndef XEMACS_CODENAME
2856 #define XEMACS_CODENAME "Noname"
2857 #endif
2858   Vxemacs_codename = Fpurecopy (build_string (XEMACS_CODENAME));
2859
2860   DEFVAR_BOOL ("noninteractive", &noninteractive1 /*
2861 Non-nil means XEmacs is running without interactive terminal.
2862 */ );
2863
2864   DEFVAR_BOOL ("inhibit-early-packages", &inhibit_early_packages /*
2865 Set to non-nil when the early packages should not be respected at startup.
2866 */ );
2867
2868   DEFVAR_BOOL ("inhibit-autoloads", &inhibit_autoloads /*
2869 Set to non-nil when autoloads should not be loaded at startup.
2870 */ );
2871
2872   DEFVAR_BOOL ("debug-paths", &debug_paths /*
2873 Set to non-nil when debug information about paths should be printed.
2874 */ );
2875
2876   DEFVAR_BOOL ("inhibit-site-lisp", &inhibit_site_lisp /*
2877 Set to non-nil when the site-lisp should not be searched at startup.
2878 */ );
2879 #ifdef INHIBIT_SITE_LISP
2880   inhibit_site_lisp = 1;
2881 #endif
2882
2883   DEFVAR_BOOL ("inhibit-site-modules", &inhibit_site_modules /*
2884 Set to non-nil when site-modules should not be searched at startup.
2885 */ );
2886 #ifdef INHIBIT_SITE_MODULES
2887   inhibit_site_modules = 1;
2888 #endif
2889
2890   DEFVAR_INT ("emacs-priority", &emacs_priority /*
2891 Priority for XEmacs to run at.
2892 This value is effective only if set before XEmacs is dumped,
2893 and only if the XEmacs executable is installed with setuid to permit
2894 it to change priority.  (XEmacs sets its uid back to the real uid.)
2895 Currently, you need to define SET_EMACS_PRIORITY in `config.h'
2896 before you compile XEmacs, to enable the code for this feature.
2897 */ );
2898   emacs_priority = 0;
2899
2900   DEFVAR_CONST_LISP ("internal-error-checking", &Vinternal_error_checking /*
2901 Internal error checking built-in into this instance of XEmacs.
2902 This is a list of symbols, initialized at build-time.  Legal symbols
2903 are:
2904
2905 extents         - check extents prior to each extent change;
2906 typecheck       - check types strictly, aborting in case of error;
2907 malloc          - check operation of malloc;
2908 gc              - check garbage collection;
2909 bufpos          - check buffer positions.
2910 */ );
2911   Vinternal_error_checking = Qnil;
2912 #ifdef ERROR_CHECK_EXTENTS
2913   Vinternal_error_checking = Fcons (intern ("extents"),
2914                                     Vinternal_error_checking);
2915 #endif
2916 #ifdef ERROR_CHECK_TYPECHECK
2917   Vinternal_error_checking = Fcons (intern ("typecheck"),
2918                                     Vinternal_error_checking);
2919 #endif
2920 #ifdef ERROR_CHECK_MALLOC
2921   Vinternal_error_checking = Fcons (intern ("malloc"),
2922                                     Vinternal_error_checking);
2923 #endif
2924 #ifdef ERROR_CHECK_GC
2925   Vinternal_error_checking = Fcons (intern ("gc"),
2926                                     Vinternal_error_checking);
2927 #endif
2928 #ifdef ERROR_CHECK_BUFPOS
2929   Vinternal_error_checking = Fcons (intern ("bufpos"),
2930                                     Vinternal_error_checking);
2931 #endif
2932   Vinternal_error_checking = Fpurecopy (Vinternal_error_checking);
2933
2934   DEFVAR_LISP ("path-separator", &Vpath_separator /*
2935 The directory separator in search paths, as a string.
2936 */ );
2937   {
2938     char c = SEPCHAR;
2939     Vpath_separator = make_string ((Bufbyte *)&c, 1);
2940   }
2941 }
2942
2943 void
2944 complex_vars_of_emacs (void)
2945 {
2946   /* This is all related to path searching. */
2947
2948   DEFVAR_LISP ("emacs-program-name", &Vemacs_program_name /*
2949 *Name of the Emacs variant.
2950 For example, this may be \"xemacs\" or \"infodock\".
2951 This is mainly meant for use in path searching.
2952 */ );
2953   Vemacs_program_name = build_string ((char *) PATH_PROGNAME);
2954
2955   DEFVAR_LISP ("emacs-program-version", &Vemacs_program_version /*
2956 *Version of the Emacs variant.
2957 This typically has the form XX.XX[-bXX].
2958 This is mainly meant for use in path searching.
2959 */ );
2960   Vemacs_program_version = build_string ((char *) PATH_VERSION);
2961
2962   DEFVAR_LISP ("exec-path", &Vexec_path /*
2963 *List of directories to search programs to run in subprocesses.
2964 Each element is a string (directory name) or nil (try default directory).
2965 */ );
2966   Vexec_path = Qnil;
2967
2968   DEFVAR_LISP ("exec-directory", &Vexec_directory /*
2969 *Directory of architecture-dependent files that come with XEmacs,
2970 especially executable programs intended for XEmacs to invoke.
2971 */ );
2972   Vexec_directory = Qnil;
2973
2974   DEFVAR_LISP ("configure-exec-directory", &Vconfigure_exec_directory /*
2975 For internal use by the build procedure only.
2976 configure's idea of what EXEC-DIRECTORY will be.
2977 */ );
2978 #ifdef PATH_EXEC
2979   Vconfigure_exec_directory = Ffile_name_as_directory
2980     (build_string ((char *) PATH_EXEC));
2981 #else
2982   Vconfigure_exec_directory = Qnil;
2983 #endif
2984
2985   DEFVAR_LISP ("lisp-directory", &Vlisp_directory /*
2986 *Directory of core Lisp files that come with XEmacs.
2987 */ );
2988   Vlisp_directory = Qnil;
2989
2990   DEFVAR_LISP ("configure-lisp-directory", &Vconfigure_lisp_directory /*
2991 For internal use by the build procedure only.
2992 configure's idea of what LISP-DIRECTORY will be.
2993 */ );
2994 #ifdef PATH_LOADSEARCH
2995   Vconfigure_lisp_directory = Ffile_name_as_directory
2996     (build_string ((char *) PATH_LOADSEARCH));
2997 #else
2998   Vconfigure_lisp_directory = Qnil;
2999 #endif
3000
3001   DEFVAR_LISP ("module-directory", &Vmodule_directory /*
3002 *Directory of core dynamic modules that come with XEmacs.
3003 */ );
3004   Vmodule_directory = Qnil;
3005
3006   DEFVAR_LISP ("configure-module-directory", &Vconfigure_module_directory /*
3007 For internal use by the build procedure only.
3008 configure's idea of what MODULE-DIRECTORY will be.
3009 */ );
3010 #ifdef PATH_MODULESEARCH
3011   Vconfigure_module_directory = Ffile_name_as_directory
3012     (build_string ((char *) PATH_MODULESEARCH));
3013 #else
3014   Vconfigure_module_directory = Qnil;
3015 #endif
3016
3017   DEFVAR_LISP ("configure-package-path", &Vconfigure_package_path /*
3018 For internal use by the build procedure only.
3019 configure's idea of what the package path will be.
3020 */ );
3021 #ifdef PATH_PACKAGEPATH
3022   Vconfigure_package_path = decode_path (PATH_PACKAGEPATH);
3023 #else
3024   Vconfigure_package_path = Qnil;
3025 #endif
3026
3027   DEFVAR_LISP ("data-directory", &Vdata_directory /*
3028 *Directory of architecture-independent files that come with XEmacs,
3029 intended for XEmacs to use.
3030 Use of this variable in new code is almost never correct.  See the
3031 function `locate-data-directory' and the variable `data-directory-list'.
3032 */ );
3033   Vdata_directory = Qnil;
3034
3035   DEFVAR_LISP ("configure-data-directory", &Vconfigure_data_directory /*
3036 For internal use by the build procedure only.
3037 configure's idea of what DATA-DIRECTORY will be.
3038 */ );
3039 #ifdef PATH_DATA
3040   Vconfigure_data_directory = Ffile_name_as_directory
3041     (build_string ((char *) PATH_DATA));
3042 #else
3043   Vconfigure_data_directory = Qnil;
3044 #endif
3045
3046   DEFVAR_LISP ("data-directory-list", &Vdata_directory_list /*
3047 *List of directories of architecture-independent files that come with XEmacs
3048 or were installed as packages, and are intended for XEmacs to use.
3049 */ );
3050   Vdata_directory_list = Qnil;
3051
3052 #ifdef CLASH_DETECTION
3053   DEFVAR_LISP ("configure-lock-directory", &Vconfigure_lock_directory /*
3054 For internal use by the build procedure only.
3055 configure's idea of what LOCK-DIRECTORY will be.
3056 */ );
3057 #ifdef PATH_LOCK
3058   Vconfigure_lock_directory = Ffile_name_as_directory
3059     (build_string ((char *) PATH_LOCK));
3060 #else
3061   Vconfigure_lock_directory = Qnil;
3062 #endif
3063 #endif /* CLASH_DETECTION */
3064
3065   DEFVAR_LISP ("site-directory", &Vsite_directory /*
3066 *Directory of site-specific Lisp files that come with XEmacs.
3067 */ );
3068   Vsite_directory = Qnil;
3069
3070   DEFVAR_LISP ("configure-site-directory", &Vconfigure_site_directory /*
3071 For internal use by the build procedure only.
3072 configure's idea of what SITE-DIRECTORY will be.
3073 */ );
3074 #ifdef PATH_SITE
3075   Vconfigure_site_directory = Ffile_name_as_directory
3076     (build_string ((char *) PATH_SITE));
3077 #else
3078   Vconfigure_site_directory = Qnil;
3079 #endif
3080
3081   DEFVAR_LISP ("site-module-directory", &Vsite_module_directory /*
3082 *Directory of site-specific loadable modules that come with XEmacs.
3083 */ );
3084   Vsite_module_directory = Qnil;
3085
3086   DEFVAR_LISP ("configure-site-module-directory", &Vconfigure_site_module_directory /*
3087 For internal use by the build procedure only.
3088 configure's idea of what SITE-DIRECTORY will be.
3089 */ );
3090 #ifdef PATH_SITE_MODULES
3091   Vconfigure_site_module_directory = Ffile_name_as_directory
3092     (build_string ((char *) PATH_SITE_MODULES));
3093 #else
3094   Vconfigure_site_module_directory = Qnil;
3095 #endif
3096
3097   DEFVAR_LISP ("doc-directory", &Vdoc_directory /*
3098 *Directory containing the DOC file that comes with XEmacs.
3099 This is usually the same as exec-directory.
3100 */ );
3101   Vdoc_directory = Qnil;
3102
3103   DEFVAR_LISP ("configure-doc-directory", &Vconfigure_doc_directory /*
3104 For internal use by the build procedure only.
3105 configure's idea of what DOC-DIRECTORY will be.
3106 */ );
3107 #ifdef PATH_DOC
3108   Vconfigure_doc_directory = Ffile_name_as_directory
3109     (build_string ((char *) PATH_DOC));
3110 #else
3111   Vconfigure_doc_directory = Qnil;
3112 #endif
3113
3114   DEFVAR_LISP ("configure-exec-prefix-directory", &Vconfigure_exec_prefix_directory /*
3115 For internal use by the build procedure only.
3116 configure's idea of what EXEC-PREFIX-DIRECTORY will be.
3117 */ );
3118 #ifdef PATH_EXEC_PREFIX
3119   Vconfigure_exec_prefix_directory = Ffile_name_as_directory
3120     (build_string ((char *) PATH_EXEC_PREFIX));
3121 #else
3122   Vconfigure_exec_prefix_directory = Qnil;
3123 #endif
3124
3125   DEFVAR_LISP ("configure-prefix-directory", &Vconfigure_prefix_directory /*
3126 For internal use by the build procedure only.
3127 configure's idea of what PREFIX-DIRECTORY will be.
3128 */ );
3129 #ifdef PATH_PREFIX
3130   Vconfigure_prefix_directory = Ffile_name_as_directory
3131     (build_string ((char *) PATH_PREFIX));
3132 #else
3133   Vconfigure_prefix_directory = Qnil;
3134 #endif
3135
3136   DEFVAR_LISP ("configure-info-directory", &Vconfigure_info_directory /*
3137 For internal use by the build procedure only.
3138 This is the name of the directory in which the build procedure installed
3139 Emacs's info files; the default value for Info-default-directory-list
3140 includes this.
3141 */ );
3142 #ifdef PATH_INFO
3143   Vconfigure_info_directory =
3144     Ffile_name_as_directory (build_string (PATH_INFO));
3145 #else
3146   Vconfigure_info_directory = Qnil;
3147 #endif
3148
3149   DEFVAR_LISP ("configure-info-path", &Vconfigure_info_path /*
3150 The configured initial path for info documentation.
3151 */ );
3152 #ifdef PATH_INFOPATH
3153   Vconfigure_info_path = decode_path (PATH_INFOPATH);
3154 #else
3155   Vconfigure_info_path = Qnil;
3156 #endif
3157 }
3158
3159 #ifdef __sgi
3160 /* This is so tremendously ugly I'd puke. But then, it works.
3161  * The target is to override the static constructor from the
3162  * libiflPNG.so library which is maskerading as libz, and
3163  * cores on us when re-started from the dumped executable.
3164  * This will have to go for 21.1  -- OG.
3165  */
3166 void __sti__iflPNGFile_c___(void);
3167 void __sti__iflPNGFile_c___()
3168 {
3169 }
3170
3171 #endif