(<DENTISTRY SYMBOL *>): Add missing `general-category'.
[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    Copyright (C) 2000, 2002 Ben Wing.
6    Copyright (C) 2000,2002 MORIOKA Tomohiko.
7
8 This file is part of XEmacs.
9
10 XEmacs is free software; you can redistribute it and/or modify it
11 under the terms of the GNU General Public License as published by the
12 Free Software Foundation; either version 2, or (at your option) any
13 later version.
14
15 XEmacs is distributed in the hope that it will be useful, but WITHOUT
16 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
17 FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
18 for more details.
19
20 You should have received a copy of the GNU General Public License
21 along with XEmacs; see the file COPYING.  If not, write to
22 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
23 Boston, MA 02111-1307, USA.  */
24
25 /* Synched up with: Mule 2.0, FSF 19.28. */
26
27 /* Capsule summary of the various releases of Lucid Emacs/XEmacs and
28    FSF/GNU Emacs.  Provided here for use in cross-referencing version
29    releases and dates in comments, esp. in the authorship comments at
30    the beginning of each file.  More information about history can be
31    found in the beginning of the Internals Manual and in the About page.
32
33
34 -- A time line for Lucid Emacs/XEmacs is
35
36 version 19.0 shipped with Energize 1.0, April 1992.
37 version 19.1 released June 4, 1992.
38 version 19.2 released June 19, 1992.
39 version 19.3 released September 9, 1992.
40 version 19.4 released January 21, 1993.
41 version 19.5 was a repackaging of 19.4 with a few bug fixes and
42   shipped with Energize 2.0.  Never released to the net.
43 version 19.6 released April 9, 1993.
44 version 19.7 was a repackaging of 19.6 with a few bug fixes and
45   shipped with Energize 2.1.  Never released to the net.
46 version 19.8 released September 6, 1993.
47 version 19.9 released January 12, 1994.
48 version 19.10 released May 27, 1994.
49 version 19.11 (first XEmacs) released September 13, 1994.
50 version 19.12 released June 23, 1995.
51 version 19.13 released September 1, 1995.
52 version 19.14 released June 23, 1996.
53 version 20.0 released February 9, 1997.
54 version 19.15 released March 28, 1997.
55 version 20.1 (not released to the net) April 15, 1997.
56 version 20.2 released May 16, 1997.
57 version 19.16 released October 31, 1997.
58 version 20.3 (the first stable version of XEmacs 20.x) released
59   November 30, 1997.
60 version 20.4 released February 28, 1998.
61
62
63 -- A time line for GNU Emacs version 19 is
64
65 version 19.7 (beta) (first beta release) released ??????; prob. late May 1993.
66 version 19.8 (beta) released May 27, 1993.
67 version 19.9 (beta) released May 27, 1993.
68 version 19.10 (beta) released May 30, 1993.
69 version 19.11 (beta) released June 1, 1993.
70 version 19.12 (beta) released June 2, 1993.
71 version 19.13 (beta) released June 8, 1993.
72 version 19.14 (beta) released June 17, 1993.
73 version 19.15 (beta) released June 19, 1993.
74 version 19.16 (beta) released July 6, 1993.
75 version 19.17 (beta) released late July, 1993.
76 version 19.18 (beta) released August 9, 1993.
77 version 19.19 (beta) released August 15, 1993.
78 version 19.20 (beta) released November 17, 1993.
79 version 19.21 (beta) released November 17, 1993.
80 version 19.22 (beta) released November 28, 1993.
81 version 19.23 (beta) released May 17, 1994.
82 version 19.24 (beta) released May 16, 1994.
83 version 19.25 (beta) released June 3, 1994.
84 version 19.26 (beta) released September 11, 1994.
85 version 19.27 (beta) released September 14, 1994.
86 version 19.28 (first ``official'' release) released November 1, 1994.
87 version 19.29 released June 21, 1995.
88 version 19.30 released November 24, 1995.
89 version 19.31 released May 25, 1996.
90 version 19.32 released July 31, 1996.
91 version 19.33 released August 11, 1996.
92 version 19.34 released August 21, 1996.
93 version 19.34b released September 6, 1996.
94
95
96 -- A time line for GNU Emacs version 20 is
97
98 version 20.1 released September 17, 1997.
99 version 20.2 released September 20, 1997.
100 version 20.3 released August 19, 1998.
101
102
103 -- A time line for GNU Emacs version 18 and older is
104
105 GNU Emacs version 15 (15.34) was released sometime in 1984 or 1985 and
106   shared some code with a version of Emacs written by James Gosling (the
107   same James Gosling who later created the Java language).
108 GNU Emacs version 16 (first released version was 16.56) was released on
109   July 15, 1985.  All Gosling code was removed due to potential copyright
110   problems with the code.
111 version 16.57: released on September 16, 1985.
112 versions 16.58, 16.59: released on September 17, 1985.
113 version 16.60: released on September 19, 1985.  These later version 16's
114   incorporated patches from the net, esp. for getting Emacs to work under
115   System V.
116 version 17.36 (first official v17 release) released on December 20, 1985.
117   Included a TeX-able user manual.  First official unpatched version that
118    worked on vanilla System V machines.
119 version 17.43 (second official v17 release) released on January 25, 1986.
120 version 17.45 released on January 30, 1986.
121 version 17.46 released on February 4, 1986.
122 version 17.48 released on February 10, 1986.
123 version 17.49 released on February 12, 1986.
124 version 17.55 released on March 18, 1986.
125 version 17.57 released on March 27, 1986.
126 version 17.58 released on April 4, 1986.
127 version 17.61 released on April 12, 1986.
128 version 17.63 released on May 7, 1986.
129 version 17.64 released on May 12, 1986.
130 version 18.24 (a beta version) released on October 2, 1986.
131 version 18.30 (a beta version) released on November 15, 1986.
132 version 18.31 (a beta version) released on November 23, 1986.
133 version 18.32 (a beta version) released on December 7, 1986.
134 version 18.33 (a beta version) released on December 12, 1986.
135 version 18.35 (a beta version) released on January 5, 1987.
136 version 18.36 (a beta version) released on January 21, 1987.
137 January 27, 1987: The Great Usenet Renaming.  net.emacs is now comp.emacs.
138 version 18.37 (a beta version) released on February 12, 1987.
139 version 18.38 (a beta version) released on March 3, 1987.
140 version 18.39 (a beta version) released on March 14, 1987.
141 version 18.40 (a beta version) released on March 18, 1987.
142 version 18.41 (the first ``official'' release) released on March 22, 1987.
143 version 18.45 released on June 2, 1987.
144 version 18.46 released on June 9, 1987.
145 version 18.47 released on June 18, 1987.
146 version 18.48 released on September 3, 1987.
147 version 18.49 released on September 18, 1987.
148 version 18.50 released on February 13, 1988.
149 version 18.51 released on May 7, 1988.
150 version 18.52 released on September 1, 1988.
151 version 18.53 released on February 24, 1989.
152 version 18.54 released on April 26, 1989.
153 version 18.55 released on August 23, 1989.  This is the earliest version
154   that is still available by FTP.
155 version 18.56 released on January 17, 1991.
156 version 18.57 released late January, 1991.
157 version 18.58 released ?????.
158 version 18.59 released October 31, 1992.
159
160 */
161
162 /* Note: It is necessary to specify <config.h> and not "config.h" in
163    order for the --srcdir type of compilation to work properly.
164    Otherwise the config.h from the srcdir, rather than the one from
165    the build dir, will be used. */
166
167 #include <config.h>
168 #include "lisp.h"
169
170 #include "backtrace.h" /* run-emacs-from-temacs needs this */
171 #include "buffer.h"
172 #include "commands.h"
173 #include "console.h"
174 #include "process.h"
175 #include "redisplay.h"
176 #include "frame.h"
177 #include "sysdep.h"
178
179 #include "syssignal.h" /* Always include before systty.h */
180 #include "systty.h"
181 #include "sysfile.h"
182 #include "systime.h"
183
184 #ifdef CYGWIN
185 #include "syswindows.h"
186 #endif
187
188 #ifdef PDUMP
189 #include "dumper.h"
190 #endif
191
192 #ifndef SEPCHAR
193 #define SEPCHAR ':'
194 #endif
195
196 #ifdef QUANTIFY
197 #include <quantify.h>
198 #endif
199
200 #ifdef HAVE_SHLIB
201 #include "sysdll.h"
202 #endif
203
204 #if defined (HAVE_LOCALE_H) && \
205    (defined (I18N2) || defined (I18N3) || defined (I18N4))
206 #include <locale.h>
207 #endif
208
209 #ifdef TOOLTALK
210 #include TT_C_H_FILE
211 #endif
212
213 #if defined (WIN32_NATIVE)
214 #include "nt.h"
215 #endif
216
217 /* For PATH_EXEC */
218 #include <paths.h>
219
220 #if defined (HEAP_IN_DATA) && !defined(PDUMP)
221 void report_sheap_usage (int die_if_pure_storage_exceeded);
222 #endif
223
224 #if !defined (SYSTEM_MALLOC) && !defined (DOUG_LEA_MALLOC)
225 extern void *(*__malloc_hook)(size_t);
226 extern void *(*__realloc_hook)(void *, size_t);
227 extern void (*__free_hook)(void *);
228 #endif  /* not SYSTEM_MALLOC && not DOUG_LEA_MALLOC */
229
230 /* Command line args from shell, as list of strings */
231 Lisp_Object Vcommand_line_args;
232
233 /* Set nonzero after XEmacs has started up the first time.
234   Prevents reinitialization of the Lisp world and keymaps
235   on subsequent starts.  */
236 int initialized;
237
238 #ifdef DOUG_LEA_MALLOC
239 # include <malloc.h>
240 /* Preserves a pointer to the memory allocated that copies that
241    static data inside glibc's malloc.  */
242 static void *malloc_state_ptr;
243 #endif /* DOUG_LEA_MALLOC */
244
245 # ifdef REL_ALLOC
246 void r_alloc_reinit (void);
247 # endif
248
249 #ifdef HAVE_GTK
250 void console_type_create_select_gtk(void);
251 #endif
252
253 /* Variable whose value is symbol giving operating system type. */
254 Lisp_Object Vsystem_type;
255
256 /* Variable whose value is string giving configuration built for.  */
257 Lisp_Object Vsystem_configuration;
258
259 /* Variable whose value is string containing the configuration options
260    XEmacs was built with.  */
261 Lisp_Object Vsystem_configuration_options;
262
263 /* Version numbers and strings */
264 Lisp_Object Vemacs_major_version;
265 Lisp_Object Vemacs_minor_version;
266 Lisp_Object Vemacs_patch_level;
267 Lisp_Object Vemacs_beta_version;
268 Lisp_Object Vxemacs_codename;
269 #ifdef INFODOCK
270 Lisp_Object Vinfodock_major_version;
271 Lisp_Object Vinfodock_minor_version;
272 Lisp_Object Vinfodock_build_version;
273 #endif
274
275 /* The path under which XEmacs was invoked. */
276 Lisp_Object Vinvocation_path;
277
278 /* The name under which XEmacs was invoked, with any leading directory
279    names discarded.  */
280 Lisp_Object Vinvocation_name;
281
282 /* The directory name from which XEmacs was invoked.  */
283 Lisp_Object Vinvocation_directory;
284
285 #if 0 /* FSFmacs */
286 /* The directory name in which to find subdirs such as lisp and etc.
287    nil means get them only from PATH_LOADSEARCH.  */
288 Lisp_Object Vinstallation_directory;
289 #endif
290
291 Lisp_Object Vemacs_program_name, Vemacs_program_version;
292 Lisp_Object Vexec_path;
293 Lisp_Object Vexec_directory, Vconfigure_exec_directory;
294 Lisp_Object Vlisp_directory, Vconfigure_lisp_directory;
295 Lisp_Object Vmule_lisp_directory, Vconfigure_mule_lisp_directory;
296 Lisp_Object Vutf_2000_lisp_directory, Vconfigure_utf_2000_lisp_directory;
297 Lisp_Object Vmodule_directory, Vconfigure_module_directory;
298 Lisp_Object Vsite_module_directory, Vconfigure_site_module_directory;
299 Lisp_Object Vconfigure_package_path;
300 Lisp_Object Vdata_directory, Vconfigure_data_directory;
301 Lisp_Object Vdoc_directory, Vconfigure_doc_directory;
302 Lisp_Object Vconfigure_lock_directory;
303 Lisp_Object Vdata_directory_list;
304 Lisp_Object Vconfigure_info_directory;
305 Lisp_Object Vsite_directory, Vconfigure_site_directory;
306 Lisp_Object Vconfigure_info_path;
307 Lisp_Object Vinternal_error_checking;
308 Lisp_Object Vmail_lock_methods, Vconfigure_mail_lock_method;
309 Lisp_Object Vpath_separator;
310
311 /* The default base directory XEmacs is installed under. */
312 Lisp_Object Vconfigure_exec_prefix_directory, Vconfigure_prefix_directory;
313
314 /* If nonzero, set XEmacs to run at this priority.  This is also used
315    in child_setup and sys_suspend to make sure subshells run at normal
316    priority. */
317 Fixnum emacs_priority;
318
319 /* Some FSF junk with running_asynch_code, to preserve the match
320    data.  Not necessary because we don't call process filters
321    asynchronously (i.e. from within QUIT). */
322 /* #### Delete this when merging the rest of my code */
323 int running_asynch_code;
324
325 /* If non-zero, a window-system was specified on the command line. */
326 int display_arg;
327
328 /* Type of display specified.  We cannot use a Lisp symbol here because
329    Lisp symbols may not initialized at the time that we set this
330    variable. */
331 const char *display_use;
332
333 /* If non-zero, then the early error handler will only print the error
334    message and exit. */
335 int suppress_early_error_handler_backtrace;
336
337 /* An address near the bottom of the stack.
338    Tells GC how to save a copy of the stack.  */
339 char *stack_bottom;
340
341 #ifdef USG_SHARED_LIBRARIES
342 /* If nonzero, this is the place to put the end of the writable segment
343    at startup.  */
344
345 uintptr_t bss_end = 0;
346 #endif
347
348 /* Number of bytes of writable memory we can expect to be able to get */
349 unsigned int lim_data;
350
351 /* WARNING!
352
353    Some LISP-visible command-line options are set by XEmacs _before_ the
354    data is dumped in building a --pdump XEmacs, but used _after_ it is
355    restored in normal operation.  Thus the dump-time values overwrite the
356    values XEmacs is getting at runtime.  Such variables must be saved
357    before loading the dumpfile, and restored afterward.
358
359    Therefore these variables may not be initialized in vars_of_emacs().
360
361    The save/restore is done immediately before and after pdump_load() in
362    main_1().  See that function for the current list of protected variables.
363
364    Note that saving/restoring is only necessary for a few variables that are
365      o command line arguments effective at runtime (as opposed to dump-time),
366      o parsed before pdump_load, and
367      o exported to Lisp via a DEFVAR.
368 */
369
370 /* Nonzero means running XEmacs without interactive terminal.  */
371
372 int noninteractive;
373
374 /* Value of Lisp variable `noninteractive'.
375    Normally same as C variable `noninteractive'
376    but nothing terrible happens if user sets this one.
377
378    Shadowed from the pdumper by `noninteractive'. */
379
380 int noninteractive1;
381
382 /* Nonzero means don't perform site-lisp searches at startup */
383 int inhibit_site_lisp;
384
385 /* Nonzero means don't perform site-modules searches at startup */
386 int inhibit_site_modules;
387
388 /* Nonzero means don't respect early packages at startup */
389 int inhibit_early_packages;
390
391 /* Nonzero means don't load package autoloads at startup */
392 int inhibit_autoloads;
393
394 /* Nonzero means don't load the dump file (ignored if not PDUMP)  */
395
396 int nodumpfile;
397
398 /* Nonzero means print debug information about path searching */
399 int debug_paths;
400
401 /* Save argv and argc.  */
402 static Extbyte **initial_argv;  /* #### currently unused */
403 static int initial_argc;        /* #### currently unused */
404
405 static void sort_args (int argc, char **argv);
406
407 Lisp_Object Qkill_emacs_hook;
408 Lisp_Object Qsave_buffers_kill_emacs;
409
410 extern Lisp_Object Vlisp_EXEC_SUFFIXES;
411
412 \f
413
414 /* Ben's capsule summary about expected and unexpected exits from XEmacs.
415
416    Expected exits occur when the user directs XEmacs to exit, for example
417    by pressing the close button on the only frame in XEmacs, or by typing
418    C-x C-c.  This runs `save-buffers-kill-emacs', which saves any necessary
419    buffers, and then exits using the primitive `kill-emacs'.
420
421    However, unexpected exits occur in a few different ways:
422
423      -- a memory access violation or other hardware-generated exception
424         occurs.  This is the worst possible problem to deal with, because
425         the fault can occur while XEmacs is in any state whatsoever, even
426         quite unstable ones.  As a result, we need to be *extremely* careful
427         what we do.
428      -- we are using one X display (or if we've used more, we've closed the
429         others already), and some hardware or other problem happens and
430         suddenly we've lost our connection to the display.  In this situation,
431         things are not so dire as in the last one; our code itself isn't
432         trashed, so we can continue execution as normal, after having set
433         things up so that we can exit at the appropriate time.  Our exit
434         still needs to be of the emergency nature; we have no displays, so
435         any attempts to use them will fail.  We simply want to auto-save
436         (the single most important thing to do during shut-down), do minimal
437         cleanup of stuff that has an independent existence outside of XEmacs,
438         and exit.
439
440         Currently, both unexpected exit scenarios described above set
441         preparing_for_armageddon to indicate that nonessential and possibly
442         dangerous things should not be done, specifically:
443
444         -- no garbage collection.
445         -- no hooks are run.
446         -- no messages of any sort from autosaving.
447         -- autosaving tries harder, ignoring certain failures.
448         -- existing frames are not deleted.
449
450         (Also, all places that set preparing_for_armageddon also
451         set dont_check_for_quit.  This happens separately because it's
452         also necessary to set other variables to make absolutely sure
453         no quitting happens.)
454
455         In the first scenario above (the access violation), we also set
456         fatal_error_in_progress.  This causes more things to not happen:
457
458         -- assertion failures do not abort.
459         -- printing code does not do code conversion or gettext when
460            printing to stdout/stderr.
461 */
462
463 /* Nonzero if handling a fatal error already. */
464 int fatal_error_in_progress;
465
466 /* Non-nil means we're going down, so we better not run any hooks
467    or do other non-essential stuff. */
468 int preparing_for_armageddon;
469
470 /* Nonzero means we're in an unstable situation and need to skip
471    i18n conversions and such during printing. */
472 int inhibit_non_essential_printing_operations;
473
474 static JMP_BUF run_temacs_catch;
475
476 static int run_temacs_argc;
477 static char **run_temacs_argv;
478 static char *run_temacs_args;
479 static EMACS_INT run_temacs_argv_size;
480 static EMACS_INT run_temacs_args_size;
481
482 static void shut_down_emacs (int sig, Lisp_Object stuff, int no_auto_save);
483
484 /* ------------------------------- */
485 /*  low-level debugging functions  */
486 /* ------------------------------- */
487
488 #if defined (WIN32_NATIVE) && defined (DEBUG_XEMACS)
489 #define debugging_breakpoint() DebugBreak ()
490 #else
491 #define debugging_breakpoint()
492 #endif
493
494 void
495 debug_break (void)
496 {
497   debugging_breakpoint ();
498 }
499
500 #if defined (WIN32_NATIVE) || defined (CYGWIN)
501
502 /* Return whether all bytes in the specified memory block can be read. */
503 int
504 debug_can_access_memory (void *ptr, Bytecount len)
505 {
506   return !IsBadReadPtr (ptr, len);
507 }
508
509 #else /* !(defined (WIN32_NATIVE) || defined (CYGWIN)) */
510
511 /* #### There must be a better way!!!! */
512
513 static JMP_BUF memory_error_jump;
514
515 static SIGTYPE
516 debug_memory_error (int signum)
517 {
518   EMACS_REESTABLISH_SIGNAL (signum, debug_memory_error);
519   EMACS_UNBLOCK_SIGNAL (signum);
520   LONGJMP (memory_error_jump, 1);
521 }
522
523 /* Return whether all bytes in the specified memory block can be read. */
524 int
525 debug_can_access_memory (void *ptr, Bytecount len)
526 {
527   /* Use volatile to protect variables from being clobbered by longjmp. */
528   SIGTYPE (*volatile old_sigbus) (int);
529   SIGTYPE (*volatile old_sigsegv) (int);
530   volatile int old_errno = errno;
531   volatile int retval = 1;
532
533   if (!SETJMP (memory_error_jump))
534     {
535       old_sigbus =
536         (SIGTYPE (*) (int)) signal (SIGBUS, debug_memory_error);
537       old_sigsegv =
538         (SIGTYPE (*) (int)) signal (SIGSEGV, debug_memory_error);
539
540       if (len > 1)
541         /* If we can, try to avoid problems with super-optimizing compilers
542            that might decide that memcmp (ptr, ptr, len) can be optimized
543            away since its result is always 1. */
544         memcmp (ptr, (char *) ptr + 1, len - 1);
545       else
546         memcmp (ptr, ptr, len);
547     }
548   else
549     retval = 0;
550   signal (SIGBUS, old_sigbus);
551   signal (SIGSEGV, old_sigsegv);
552   errno = old_errno;
553
554   return retval;
555 }
556
557 #endif /* defined (WIN32_NATIVE) || defined (CYGWIN) */
558
559 #ifdef DEBUG_XEMACS
560
561 DEFUN ("force-debugging-signal", Fforce_debugging_signal, 0, 1, 0, /*
562 Cause XEmacs to enter the debugger.
563 On some systems, there may be no way to do this gracefully; if so,
564 nothing happens unless ABORT is non-nil, in which case XEmacs will
565 abort() -- a sure-fire way to immediately get back to the debugger,
566 but also a sure-fire way to kill XEmacs (and dump core on Unix
567 systems)!
568 */
569        (abort_))
570 {
571   debugging_breakpoint ();
572   if (!NILP (abort_))
573     abort ();
574   return Qnil;
575 }
576
577 #endif /* DEBUG_XEMACS */
578
579 static void
580 ensure_no_quitting_from_now_on (void)
581 {
582   /* make sure no quitting from now on!! */
583   dont_check_for_quit = 1;
584   Vinhibit_quit = Qt;
585   Vquit_flag = Qnil;
586 }
587
588 /* Handle bus errors, illegal instruction, etc. */
589 SIGTYPE
590 fatal_error_signal (int sig)
591 {
592   fatal_error_in_progress++;
593   inhibit_non_essential_printing_operations = 1;
594   preparing_for_armageddon = 1;
595
596   ensure_no_quitting_from_now_on ();
597
598   /* Unblock the signal so that if the same signal gets sent in the
599      code below, we avoid a deadlock. */
600   EMACS_UNBLOCK_SIGNAL (sig);
601
602   /* Only try auto-saving first time through.  If we crash in auto-saving,
603      don't do it again. */
604   if (fatal_error_in_progress == 1)
605     {
606       Fdo_auto_save (Qt, Qnil); /* do this before anything hazardous */
607       /* Do this so that the variable has the same value of 2 regardless of
608          whether we made it through auto-saving correctly. */
609       fatal_error_in_progress++;
610     }
611   else if (fatal_error_in_progress == 2)
612     stderr_out ("WARNING: Unable to auto-save your files properly.\n"
613                 "Some or all may in fact have been auto-saved.\n"
614                 "\n");
615
616   /* Now, reset our signal handler, so the next time, we just die.
617      Don't do this before auto-saving. */
618   signal (sig, SIG_DFL);
619
620   /* Keep in mind that there's more than one signal that we can crash
621      on. */
622   /* If fatal error occurs in code below, avoid infinite recursion.  */
623   if (fatal_error_in_progress <= 2)
624     {
625       shut_down_emacs (sig, Qnil, 1);
626       stderr_out ("\nLisp backtrace follows:\n\n");
627       Fbacktrace (Qexternal_debugging_output, Qt);
628 # if 0  /* This is evil, rarely useful, and causes grief in some cases. */
629       /* Check for Sun-style stack printing via /proc */
630       {
631         const char *pstack = "/usr/proc/bin/pstack";
632         if (access (pstack, X_OK) == 0)
633           {
634             char buf[100];
635             stderr_out ("\nC backtrace follows:\n"
636                        "(A real debugger may provide better information)\n\n");
637             sprintf (buf, "%s %d >&2", pstack, (int)getpid());
638             system (buf);
639           }
640       }
641 # endif
642     }
643   /* Signal the same code; this time it will really be fatal. */
644   kill (getpid (), sig);
645   SIGRETURN;
646 }
647
648 #ifdef _MSC_VER
649
650 static DWORD
651 mswindows_handle_hardware_exceptions (DWORD code)
652 {
653   if (code != STATUS_ACCESS_VIOLATION && code != STATUS_ILLEGAL_INSTRUCTION
654       && code != STATUS_PRIVILEGED_INSTRUCTION
655       && code != STATUS_DATATYPE_MISALIGNMENT)
656     return EXCEPTION_CONTINUE_SEARCH;
657
658   /* I don't know if this filter is still wrapped in the outer __try, but
659      it doesn't hurt to have another one. --ben
660      And it lets us control more exactly what we really want to do in such
661      a situation. */
662   __try
663     {
664       fatal_error_in_progress++;
665       inhibit_non_essential_printing_operations = 1;
666       preparing_for_armageddon = 1;
667
668       ensure_no_quitting_from_now_on ();
669
670       /* Only try auto-saving first time through.  If we crash in auto-saving,
671          don't do it again. */
672       if (fatal_error_in_progress == 1)
673         {
674           Fdo_auto_save (Qt, Qnil); /* do this before anything hazardous */
675           /* Do this so that the variable has the same value of 2 regardless of
676              whether we made it through auto-saving correctly. */
677           fatal_error_in_progress++;
678         }
679       else if (fatal_error_in_progress == 2)
680         stderr_out ("WARNING: Unable to auto-save your files properly.\n"
681                     "Some or all may in fact have been auto-saved.\n"
682                     "\n");
683
684       /* If fatal error occurs in code below, avoid infinite recursion.  */
685       if (fatal_error_in_progress <= 2)
686         {
687           shut_down_emacs (-1, Qnil, 1);
688           stderr_out ("\nLisp backtrace follows:\n\n");
689           Fbacktrace (Qexternal_debugging_output, Qt);
690         }
691     }
692   /* VC++ documentation says that
693      GetExceptionCode() cannot be called inside the filter itself. */
694
695   /*  __except (mswindows_handle_hardware_exceptions (GetExceptionCode ())) {}
696
697      The line above is original.  Unfortunately, when an error is tripped
698      inside of the handler (e.g. during Fbacktrace()), and the handler for
699      the handler is invoked, it correctly notices that something is amiss
700      and it should just return -- but it returns EXCEPTION_CONTINUE_SEARCH,
701      which causes the debugger to be invoked debugging the handler code in
702      this function -- and WITH THE STACK UNWOUND so that you see main()
703      calling mswindows_handle_hardware_exceptions(), calling Fbacktrace(),
704      and a crash a couple of frames in -- AND NO SIGN OF THE ORIGINAL CRASH!
705
706      There's some real weirdness going on in the stack handling -- unlike
707      in Unix, where further crashes just keep adding to the stack, it seems
708      that under the structured-exception-handling, the stack can actually
709      bounce back and forth between the full stack at the location of the
710      exception and the unwound stack at the place where the __try clause was
711      established.  I don't completely understand it.  What I do know is that
712      returning EXCEPTION_EXECUTE_HANDLER on nested crash has the effect of
713      aborting execution of the handler and going back to the outer filter
714      function, which returns EXCEPTION_CONTINUE_SEARCH and everything is
715      hunky-dorey -- your debugger sees a crash at the right location with
716      the right stack.
717
718      I'm leaving in the trickier Unix-like code in the handler; someone who
719      understands better than me how the stack works in these handlers could
720      fix it up more.  As it is, it works pretty well, so I'm not likely to
721      touch it more. --ben
722   */
723
724   __except (EXCEPTION_EXECUTE_HANDLER) {}
725
726
727   /* pretend we didn't handle this, so that the debugger is invoked and/or
728      the normal GPF box appears. */
729   return EXCEPTION_CONTINUE_SEARCH;
730 }
731
732 #endif /* _MSC_VER */
733
734 \f
735
736 #ifdef SIGDANGER
737
738 /* Handler for SIGDANGER.  */
739 SIGTYPE
740 memory_warning_signal (int sig)
741 {
742   /* #### bad bad bad; this function shouldn't do anything except
743      set a flag, or weird corruption could happen. */
744   signal (sig, memory_warning_signal);
745
746   malloc_warning
747     (GETTEXT ("Operating system warns that virtual memory is running low.\n"));
748
749   /* It might be unsafe to call do_auto_save now.  */
750   force_auto_save_soon ();
751 }
752 #endif /* SIGDANGER */
753 \f
754 /* Code for dealing with Lisp access to the Unix command line */
755
756 static Lisp_Object
757 make_arg_list_1 (int argc, Extbyte **argv, int skip_args)
758 {
759   Lisp_Object result = Qnil;
760   REGISTER int i;
761
762   for (i = argc - 1; i >= 0; i--)
763     {
764       if (i == 0 || i > skip_args)
765         {
766 #ifdef WIN32_NATIVE
767           if (i == 0)
768             {
769               /* Do not trust to what crt0 has stuffed into argv[0] */
770               char full_exe_path[MAX_PATH];
771               Lisp_Object fullpath;
772
773               GetModuleFileName (NULL, full_exe_path, MAX_PATH);
774               fullpath = build_ext_string (full_exe_path, Qmswindows_tstr);
775               result = Fcons (fullpath, result);
776 #if defined(HAVE_SHLIB)
777               {
778                 Extbyte *fullpathext;
779
780                 LISP_STRING_TO_EXTERNAL (fullpath, fullpathext,
781                                   Qdll_filename_encoding);
782                 (void) dll_init (fullpathext);
783               }
784 #endif
785             }
786           else
787 #endif
788             result = Fcons (build_ext_string (argv[i],
789                                               Qcommand_argument_encoding),
790                             result);
791         }
792     }
793   return result;
794 }
795
796 Lisp_Object
797 make_arg_list (int argc, Extbyte **argv)
798 {
799   return make_arg_list_1 (argc, argv, 0);
800 }
801
802 /* Calling functions are also responsible for calling free_argc_argv
803    when they are done with the generated list. */
804 void
805 make_argc_argv (Lisp_Object argv_list, int *argc, Extbyte ***argv)
806 {
807   Lisp_Object next;
808   int n = XINT (Flength (argv_list));
809   REGISTER int i;
810   *argv = (Extbyte**) xmalloc ((n+1) * sizeof (Extbyte*));
811
812   for (i = 0, next = argv_list; i < n; i++, next = XCDR (next))
813     {
814       const Extbyte *temp;
815       CHECK_STRING (XCAR (next));
816
817       LISP_STRING_TO_EXTERNAL (XCAR (next), temp, Qcommand_argument_encoding);
818       (*argv) [i] = xstrdup (temp);
819     }
820   (*argv) [n] = 0;
821   *argc = i;
822 }
823
824 void
825 free_argc_argv (Extbyte **argv)
826 {
827   int elt = 0;
828
829   while (argv[elt])
830     {
831       xfree (argv[elt]);
832       elt++;
833     }
834   xfree (argv);
835 }
836
837 static void
838 init_cmdargs (int argc, Extbyte **argv, int skip_args)
839 {
840   initial_argv = argv;
841   initial_argc = argc;
842
843   Vcommand_line_args = make_arg_list_1 (argc, argv, skip_args);
844 }
845
846 DEFUN ("invocation-name", Finvocation_name, 0, 0, 0, /*
847 Return the program name that was used to run XEmacs.
848 Any directory names are omitted.
849 */
850        ())
851 {
852   return Fcopy_sequence (Vinvocation_name);
853 }
854
855 DEFUN ("invocation-directory", Finvocation_directory, 0, 0, 0, /*
856 Return the directory name in which the Emacs executable was located.
857 */
858        ())
859 {
860   return Fcopy_sequence (Vinvocation_directory);
861 }
862
863 \f
864 #ifdef I18N4
865                                 /* #### - don't know why I18N4 on SunOS/JLE
866                                    can't deal with this.  It's a potential
867                                    bug that needs to be looked at. */
868 # undef RUN_TIME_REMAP
869 #endif
870
871 /* Test whether the next argument in ARGV matches SSTR or a prefix of
872    LSTR (at least MINLEN characters).  If so, then if VALPTR is non-null
873    (the argument is supposed to have a value) store in *VALPTR either
874    the next argument or the portion of this one after the equal sign.
875    ARGV is read starting at position *SKIPPTR; this index is advanced
876    by the number of arguments used.
877
878    Too bad we can't just use getopt for all of this, but we don't have
879    enough information to do it right.  */
880
881 static int
882 argmatch (char **argv, int argc, char *sstr, char *lstr,
883           int minlen, char **valptr, int *skipptr)
884 {
885   char *p = NULL;
886   int arglen;
887   char *arg;
888
889   /* Don't access argv[argc]; give up in advance.  */
890   if (argc <= *skipptr + 1)
891     return 0;
892
893   arg = argv[*skipptr+1];
894   if (arg == NULL)
895     return 0;
896   if (strcmp (arg, sstr) == 0)
897     {
898       if (valptr != NULL)
899         {
900           *valptr = argv[*skipptr+2];
901           *skipptr += 2;
902         }
903       else
904         *skipptr += 1;
905       return 1;
906     }
907   arglen = (valptr != NULL && (p = strchr (arg, '=')) != NULL
908             ? p - arg : (int) strlen (arg));
909   if (lstr == 0 || arglen < minlen || strncmp (arg, lstr, arglen) != 0)
910     return 0;
911   else if (valptr == NULL)
912     {
913       *skipptr += 1;
914       return 1;
915     }
916   else if (p != NULL)
917     {
918       *valptr = p+1;
919       *skipptr += 1;
920       return 1;
921     }
922   else if (argv[*skipptr+2] != NULL)
923     {
924       *valptr = argv[*skipptr+2];
925       *skipptr += 2;
926       return 1;
927     }
928   else
929     {
930       return 0;
931     }
932 }
933
934 /* Make stack traces always identify version + configuration */
935 #define main_1 STACK_TRACE_EYE_CATCHER
936
937 /* This function is not static, so that the compiler is less likely to
938    inline it, which would make it not show up in stack traces.
939
940    The restart argument is a flag that indicates that main_1 is now
941    being called for the second time in this invocation of xemacs; this can
942    only happen in an xemacs that is not loaded with dumped data (temacs
943    with the conventional dumper or xemacs -nd with the pdumper).   See
944    Frun_emacs_from_temacs().
945
946    restart interacts with initialized as follows (per Olivier Galibert):
947
948      It's perverted.
949
950      initialized==0 => temacs
951      initialized!=0 && restart!=0 => run-temacs
952      initialized!=0 && restart==0 => xemacs/post pdump_load()
953 */
954 DECLARE_DOESNT_RETURN (main_1 (int, char **, char **, int));
955 DOESNT_RETURN
956 main_1 (int argc, char **argv, char **envp, int restart)
957 {
958   char stack_bottom_variable;
959   int skip_args = 0;
960   Lisp_Object load_me;
961   int inhibit_window_system;
962 #ifdef NeXT
963   extern int malloc_cookie;
964 #endif
965
966 #if (!defined (SYSTEM_MALLOC) && !defined (HAVE_LIBMCHECK)      \
967      && !defined (DOUG_LEA_MALLOC))
968   /* Make sure that any libraries we link against haven't installed a
969      hook for a gmalloc of a potentially incompatible version. */
970   /* If we're using libmcheck, the hooks have already been initialized, */
971   /* don't touch them. -slb */
972   __malloc_hook = NULL;
973   __realloc_hook = NULL;
974   __free_hook = NULL;
975 #endif /* not SYSTEM_MALLOC or HAVE_LIBMCHECK or DOUG_LEA_MALLOC */
976
977   noninteractive = 0;
978   inhibit_non_essential_printing_operations = 1;
979
980 #ifdef NeXT
981   /* 19-Jun-1995 -baw
982    * NeXT secret magic, ripped from Emacs-for-NS by Carl Edman
983    * <cedman@princeton.edu>.  Note that even Carl doesn't know what this
984    * does; it was provided by NeXT, and it presumable makes NS's mallocator
985    * work with dumping.  But malloc_jumpstart() and malloc_freezedry() in
986    * unexnext.c are both completely undocumented, even in NS header files!
987    * But hey, it solves all NS related memory problems, so who's
988    * complaining? */
989   if (initialized && malloc_jumpstart (malloc_cookie) != 0)
990     stderr_out ("malloc jumpstart failed!\n");
991 #endif /* NeXT */
992
993   /*
994 #if defined (GNU_MALLOC) && \
995     defined (ERROR_CHECK_MALLOC) && \
996     !defined (HAVE_LIBMCHECK)
997   */
998 #if defined(LOSING_GCC_DESTRUCTOR_FREE_BUG)
999   /* Prior to XEmacs 21, this was `#if 0'ed out.  */
1000   /* I'm enabling this because it is the only reliable way I've found to */
1001   /* prevent a very annoying problem where GCC will attempt to free(3) */
1002   /* memory at exit() and cause a coredump. */
1003   init_free_hook ();
1004 #endif
1005
1006   sort_args (argc, argv);
1007
1008 #if defined (WIN32_NATIVE) || defined (_SCO_DS)
1009   environ = envp;
1010 #endif
1011
1012   /* Record (approximately) where the stack begins.  */
1013   stack_bottom = &stack_bottom_variable;
1014
1015 #ifdef USG_SHARED_LIBRARIES
1016   if (bss_end)
1017     brk ((void *) bss_end);
1018 #endif
1019
1020   clearerr (stdin);
1021
1022 #if defined (HAVE_MMAP) && defined (REL_ALLOC)
1023   /* ralloc can only be used if using the GNU memory allocator. */
1024   init_ralloc ();
1025 #elif defined (REL_ALLOC) && !defined(DOUG_LEA_MALLOC)
1026   if (initialized)
1027     init_ralloc();
1028 #endif
1029
1030 #ifdef HAVE_SOCKS
1031   if (initialized)
1032     SOCKSinit (argv[0]);
1033 #endif /* HAVE_SOCKS */
1034
1035 #ifndef SYSTEM_MALLOC
1036   if (!initialized)
1037     /* Arrange to get warning messages as memory fills up.  */
1038     memory_warnings (0, malloc_warning);
1039 #endif  /* not SYSTEM_MALLOC */
1040
1041 #ifdef SET_EMACS_PRIORITY
1042   if (emacs_priority != 0)
1043     nice (-emacs_priority);
1044   setuid (getuid ());
1045 #endif /* SET_EMACS_PRIORITY */
1046
1047 #ifdef EXTRA_INITIALIZE
1048   EXTRA_INITIALIZE;
1049 #endif
1050
1051 #ifdef HAVE_WINDOW_SYSTEM
1052   inhibit_window_system = 0;
1053 #else
1054   inhibit_window_system = 1;
1055 #endif
1056
1057   /* Handle the -sd/--show-dump-id switch, which means show the hex dump_id and quit */
1058   if (argmatch (argv, argc, "-sd", "--show-dump-id", 9, NULL, &skip_args))
1059     {
1060 #ifdef PDUMP
1061       printf ("%08x\n", dump_id);
1062 #else
1063       printf ("Portable dumper not configured; -sd just forces exit.\n");
1064 #endif
1065       exit (0);
1066     }
1067
1068   /* Handle the -t switch, which specifies filename to use as terminal */
1069   {
1070     char *term;
1071     if (argmatch (argv, argc, "-t", "--terminal", 4, &term, &skip_args))
1072       {
1073         close (0);
1074         close (1);
1075         if (open (term, O_RDWR | OPEN_BINARY, 2) < 0)
1076           fatal ("%s: %s", term, strerror (errno));
1077         dup (0);
1078         if (! isatty (0))
1079           fatal ("%s: not a tty", term);
1080
1081 #if 0
1082         stderr_out ("Using %s", ttyname (0));
1083 #endif
1084         stderr_out ("Using %s", term);
1085         inhibit_window_system = 1;      /* -t => -nw */
1086       }
1087   }
1088
1089   /* Handle the --no-dump-file/-nd switch, which means don't load the dump file (ignored when not using pdump) */
1090   if (argmatch (argv, argc, "-nd", "--no-dump-file", 7, NULL, &skip_args))
1091     {
1092       nodumpfile = 1;
1093     }
1094
1095   /* Handle -nw switch */
1096   if (argmatch (argv, argc, "-nw", "--no-windows", 6, NULL, &skip_args))
1097     inhibit_window_system = 1;
1098
1099   /* Handle the -batch switch, which means don't do interactive display.  */
1100   if (argmatch (argv, argc, "-batch", "--batch", 5, NULL, &skip_args))
1101     {
1102 #if 0 /* I don't think this is correct. */
1103       inhibit_autoloads = 1;
1104 #endif
1105       noninteractive = 1;
1106     }
1107
1108   if (argmatch (argv, argc, "-debug-paths", "--debug-paths",
1109                 11, NULL, &skip_args))
1110       debug_paths = 1;
1111
1112   /* Partially handle -no-autoloads, -no-early-packages and -vanilla.  Packages */
1113   /* are searched prior to the rest of the command line being parsed in */
1114   /* startup.el */
1115   if (argmatch (argv, argc, "-no-early-packages", "--no-early-packages",
1116                 6, NULL, &skip_args))
1117     {
1118       inhibit_early_packages = 1;
1119       skip_args--;
1120     }
1121 #ifdef HAVE_SHLIB
1122   if (argmatch (argv, argc, "-no-site-modules", "--no-site-modules",
1123                 9, NULL, &skip_args))
1124     {
1125       inhibit_site_modules = 1;
1126       skip_args--;
1127     }
1128 #else
1129   inhibit_site_modules = 1;
1130 #endif
1131   if (argmatch (argv, argc, "-vanilla", "--vanilla",
1132                 7, NULL, &skip_args))
1133     {
1134       inhibit_early_packages = 1;
1135       skip_args--;
1136     }
1137
1138   if (argmatch (argv, argc, "-no-autoloads", "--no-autoloads",
1139                 7, NULL, &skip_args))
1140     {
1141       /* Inhibit everything */
1142       inhibit_autoloads = 1;
1143       skip_args--;
1144     }
1145
1146   if (argmatch (argv, argc, "-debug-paths", "--debug-paths",
1147                 6, NULL, &skip_args))
1148     {
1149       debug_paths = 1;
1150       skip_args--;
1151     }
1152
1153
1154   /* Partially handle the -version and -help switches: they imply -batch,
1155      but are not removed from the list. */
1156   if (argmatch (argv, argc, "-help", "--help",   3, NULL, &skip_args))
1157     noninteractive = 1, skip_args--;
1158
1159   if (argmatch (argv, argc, "-version", "--version", 3, NULL, &skip_args) ||
1160       argmatch (argv, argc, "-V",    0,              2, NULL, &skip_args))
1161       noninteractive = 1, skip_args--;
1162
1163   /* Now, figure out which type of console is our first console. */
1164
1165   display_arg = 0;
1166
1167   if (noninteractive)
1168     display_use = "stream";
1169   else
1170     display_use = "tty";
1171
1172 #ifndef HAVE_TTY
1173   if (inhibit_window_system)
1174     fatal ("Sorry, this XEmacs was not compiled with TTY support");
1175 #endif
1176
1177 #ifdef HAVE_WINDOW_SYSTEM
1178   /* Stupid kludge to catch command-line display spec.  We can't
1179      handle this argument entirely in window-system-dependent code
1180      because we don't even know which window-system-dependent code
1181      to run until we've recognized this argument.  */
1182   if (!inhibit_window_system && !noninteractive)
1183     {
1184 #ifdef HAVE_X_WINDOWS
1185       char *dpy = 0;
1186       int count_before = skip_args;
1187
1188       if (argmatch (argv, argc, "-d", "--display", 3, &dpy, &skip_args) ||
1189           argmatch (argv, argc, "-display", 0,     3, &dpy, &skip_args))
1190         {
1191           display_arg = 1;
1192           display_use = "x";
1193         }
1194       /* If we have the form --display=NAME,
1195          convert it into  -d name.
1196          This requires inserting a new element into argv.  */
1197       if (dpy != 0 && skip_args - count_before == 1)
1198         {
1199           char **new = (char **) xmalloc (sizeof (char *) * (argc + 2));
1200           int j;
1201
1202           for (j = 0; j < count_before + 1; j++)
1203             new[j] = argv[j];
1204           new[count_before + 1] = "-d";
1205           new[count_before + 2] = dpy;
1206           for (j = count_before + 2; j <argc; j++)
1207             new[j + 1] = argv[j];
1208           argv = new;
1209           argc++;
1210         }
1211       /* Change --display to -d, when its arg is separate.  */
1212       else if (dpy != 0 && skip_args > count_before
1213                && argv[count_before + 1][1] == '-')
1214         argv[count_before + 1] = "-d";
1215
1216       /* Don't actually discard this arg.  */
1217       skip_args = count_before;
1218
1219       /* If there is a non-empty environment var DISPLAY, set
1220          `display_use', but not `display_arg', which is only to be set
1221          if the display was specified on the command line. */
1222       if ((dpy = getenv ("DISPLAY")) && dpy[0])
1223         display_use = "x";
1224
1225 #endif /* HAVE_X_WINDOWS */
1226 #ifdef HAVE_GTK
1227       {
1228         char *dpy = getenv ("DISPLAY");
1229         if (dpy && dpy[0])
1230           display_use = "gtk";
1231       }
1232 #endif
1233 #ifdef HAVE_MS_WINDOWS
1234       if (strcmp(display_use, "x") != 0)
1235         display_use = "mswindows";
1236 #endif /* HAVE_MS_WINDOWS */
1237     }
1238 #endif /* HAVE_WINDOW_SYSTEM */
1239
1240   noninteractive1 = noninteractive;
1241
1242   /****** Now initialize everything *******/
1243
1244   /* First, do really basic environment initialization -- catching signals
1245      and the like.  These functions have no dependence on any part of
1246      the Lisp engine and need to be done both at dump time and at run time. */
1247
1248   init_signals_very_early ();
1249   init_data_very_early (); /* Catch math errors. */
1250 #ifdef LISP_FLOAT_TYPE
1251   init_floatfns_very_early (); /* Catch floating-point math errors. */
1252 #endif
1253   init_process_times_very_early (); /* Initialize our process timers.
1254                                        As early as possible, of course,
1255                                        so we can be fairly accurate. */
1256   init_intl_very_early (); /* set up the locale and domain for gettext and
1257                               such. */
1258 #ifdef HAVE_MS_WINDOWS
1259   init_mswindows_very_early (); /* Some things - like dde need to be
1260                                    initialized early so that the
1261                                    client doesn't give up waiting.  */
1262 #endif
1263
1264   /* Now initialize the Lisp engine and the like.  Done only during
1265      dumping.  No dependence on anything that may be in the user's
1266      environment when the dumped XEmacs is run.
1267
1268      We try to do things in an order that minimizes the non-obvious
1269      dependencies between functions. */
1270
1271   /* purify_flag 1 is correct even if CANNOT_DUMP.
1272    * loadup.el will set to nil at end. */
1273
1274   purify_flag = 0;
1275 #ifdef PDUMP
1276   if (restart)
1277     initialized = 1;
1278   else if (nodumpfile) {
1279     initialized = 0;
1280     purify_flag = 1;
1281   } else {
1282
1283     /* Keep command options from getting stomped.
1284
1285        Some LISP-visible options are changed by XEmacs _after_ the data is
1286        dumped in building a --pdump XEmacs, but _before_ it is restored in
1287        normal operation.  Thus the restored values overwrite the values
1288        XEmacs is getting at run-time.  Such variables must be saved here,
1289        and restored after loading the dumped data.
1290
1291        Boy, this is ugly, but how else to do it?
1292     */
1293
1294     /* noninteractive1 is saved in noninteractive, which isn't LISP-visible */
1295     int inhibit_early_packages_save = inhibit_early_packages;
1296     int inhibit_autoloads_save      = inhibit_autoloads;
1297     int debug_paths_save            = debug_paths;
1298     /* #### Give inhibit-site-lisp a command switch?  If so, uncomment: */
1299     /* int inhibit_site_lisp_save      = inhibit_site_lisp; */
1300     int inhibit_site_modules_save   = inhibit_site_modules;
1301
1302     initialized = pdump_load (argv[0]);
1303
1304     /* Now unstomp everything */
1305     noninteractive1        = noninteractive;
1306     inhibit_early_packages = inhibit_early_packages_save;
1307     inhibit_autoloads      = inhibit_autoloads_save;
1308     debug_paths            = debug_paths_save;
1309     /* #### Give inhibit-site-lisp a command switch?  If so, uncomment: */
1310     /* inhibit_site_lisp      = inhibit_site_lisp_save; */
1311     inhibit_site_modules   = inhibit_site_modules_save;
1312
1313     if (initialized)
1314       run_temacs_argc = -1;
1315     else
1316       purify_flag = 1;
1317   }
1318 #else
1319   if (!initialized)
1320     purify_flag = 1;
1321 #endif
1322
1323   if (!initialized)
1324     {
1325       /* Initialize things so that new Lisp objects
1326          can be created and objects can be staticpro'd.
1327          Must be basically the very first thing done
1328          because pretty much all of the initialization
1329          routines below create new objects. */
1330       init_alloc_once_early ();
1331
1332       /* Initialize Qnil, Qt, Qunbound, and the
1333          obarray.  After this, symbols can be
1334          interned.  This depends on init_alloc_once_early(). */
1335       init_symbols_once_early ();
1336
1337       /* Declare the basic symbols pertaining to errors,
1338          So that DEFERROR*() can be called. */
1339       init_errors_once_early ();
1340
1341       /* Make sure that opaque pointers can be created. */
1342       init_opaque_once_early ();
1343
1344       /* Now declare all the symbols and define all the Lisp primitives.
1345
1346          The *only* thing that the syms_of_*() functions are allowed to do
1347          is call one of the following:
1348
1349          INIT_LRECORD_IMPLEMENTATION()
1350          defsymbol(), DEFSYMBOL(), or DEFSYMBOL_MULTIWORD_PREDICATE()
1351          defsubr() (i.e. DEFSUBR)
1352          deferror(), DEFERROR(), or DEFERROR_STANDARD()
1353          defkeyword() or DEFKEYWORD()
1354
1355          Order does not matter in these functions.
1356          */
1357
1358       syms_of_abbrev ();
1359       syms_of_alloc ();
1360       syms_of_buffer ();
1361       syms_of_bytecode ();
1362       syms_of_callint ();
1363       syms_of_callproc ();
1364       syms_of_casefiddle ();
1365       syms_of_casetab ();
1366       syms_of_chartab ();
1367       syms_of_cmdloop ();
1368       syms_of_cmds ();
1369       syms_of_console ();
1370       syms_of_data ();
1371 #ifdef DEBUG_XEMACS
1372       syms_of_debug ();
1373       syms_of_tests ();
1374 #endif /* DEBUG_XEMACS */
1375       syms_of_device ();
1376 #ifdef HAVE_DIALOGS
1377       syms_of_dialog ();
1378 #endif
1379       syms_of_dired ();
1380       syms_of_doc ();
1381       syms_of_editfns ();
1382       syms_of_elhash ();
1383       syms_of_emacs ();
1384       syms_of_eval ();
1385 #ifdef HAVE_X_WINDOWS
1386       syms_of_event_Xt ();
1387 #endif
1388 #ifdef HAVE_GTK
1389       syms_of_event_gtk ();
1390 #endif
1391 #ifdef HAVE_DRAGNDROP
1392       syms_of_dragdrop ();
1393 #endif
1394       syms_of_event_stream ();
1395       syms_of_events ();
1396       syms_of_extents ();
1397       syms_of_faces ();
1398       syms_of_fileio ();
1399 #ifdef CLASH_DETECTION
1400       syms_of_filelock ();
1401 #endif /* CLASH_DETECTION */
1402       syms_of_floatfns ();
1403       syms_of_fns ();
1404       syms_of_font_lock ();
1405       syms_of_frame ();
1406       syms_of_general ();
1407       syms_of_glyphs ();
1408       syms_of_glyphs_eimage ();
1409       syms_of_glyphs_widget ();
1410       syms_of_gui ();
1411       syms_of_gutter ();
1412       syms_of_indent ();
1413       syms_of_intl ();
1414       syms_of_keymap ();
1415       syms_of_lread ();
1416       syms_of_macros ();
1417       syms_of_marker ();
1418       syms_of_md5 ();
1419 #ifdef HAVE_DATABASE
1420       syms_of_database ();
1421 #endif
1422 #ifdef HAVE_MENUBARS
1423       syms_of_menubar ();
1424 #endif
1425       syms_of_minibuf ();
1426 #ifdef HAVE_SHLIB
1427       syms_of_module ();
1428 #endif
1429       syms_of_objects ();
1430       syms_of_print ();
1431 #if !defined (NO_SUBPROCESSES)
1432       syms_of_process ();
1433 #ifdef HAVE_WIN32_PROCESSES
1434       syms_of_process_nt ();
1435 #endif
1436 #endif
1437       syms_of_profile ();
1438 #if defined (HAVE_MMAP) && defined (REL_ALLOC) && !defined(DOUG_LEA_MALLOC)
1439       syms_of_ralloc ();
1440 #endif /* HAVE_MMAP && REL_ALLOC */
1441       syms_of_rangetab ();
1442       syms_of_redisplay ();
1443       syms_of_search ();
1444       syms_of_select ();
1445       syms_of_signal ();
1446       syms_of_sound ();
1447       syms_of_specifier ();
1448       syms_of_symbols ();
1449       syms_of_syntax ();
1450 #ifdef HAVE_SCROLLBARS
1451       syms_of_scrollbar ();
1452 #endif
1453 #ifdef HAVE_TOOLBARS
1454       syms_of_toolbar ();
1455 #endif
1456       syms_of_undo ();
1457       syms_of_widget ();
1458       syms_of_window ();
1459
1460 #ifdef HAVE_TTY
1461       syms_of_console_tty ();
1462       syms_of_device_tty ();
1463       syms_of_objects_tty ();
1464 #endif
1465
1466 #ifdef HAVE_GTK
1467       syms_of_device_gtk ();
1468       syms_of_frame_gtk ();
1469       syms_of_glyphs_gtk ();
1470       syms_of_objects_gtk ();
1471       syms_of_ui_gtk ();
1472       syms_of_select_gtk ();
1473 #ifdef HAVE_DIALOGS
1474       syms_of_dialog_gtk ();
1475 #endif
1476 #ifdef HAVE_MENUBARS
1477       syms_of_menubar_gtk ();
1478 #endif
1479       syms_of_select_gtk ();
1480       
1481 #if defined (HAVE_MENUBARS) || defined(HAVE_SCROLLBARS) || defined(HAVE_DIALOGS) || defined(HAVE_TOOLBARS)
1482       syms_of_gui_gtk ();
1483 #endif
1484 #endif /* HAVE_GTK */
1485
1486 #ifdef HAVE_X_WINDOWS
1487 #ifdef HAVE_BALLOON_HELP
1488       syms_of_balloon_x ();
1489 #endif
1490       syms_of_device_x ();
1491 #ifdef HAVE_DIALOGS
1492       syms_of_dialog_x ();
1493 #endif
1494       syms_of_frame_x ();
1495       syms_of_glyphs_x ();
1496       syms_of_objects_x ();
1497 #ifdef HAVE_MENUBARS
1498       syms_of_menubar_x ();
1499 #endif
1500       syms_of_select_x ();
1501 #if defined (HAVE_MENUBARS) || defined (HAVE_SCROLLBARS) || defined (HAVE_DIALOGS) || defined (HAVE_TOOLBARS)
1502       syms_of_gui_x ();
1503 #endif
1504 #ifdef HAVE_XIM
1505 #ifdef XIM_XLIB
1506       syms_of_input_method_xlib ();
1507 #endif
1508 #endif /* HAVE_XIM */
1509 #endif /* HAVE_X_WINDOWS */
1510
1511 #ifdef HAVE_MS_WINDOWS
1512       syms_of_console_mswindows ();
1513       syms_of_device_mswindows ();
1514       syms_of_dialog_mswindows ();
1515       syms_of_frame_mswindows ();
1516       syms_of_objects_mswindows ();
1517       syms_of_select_mswindows ();
1518       syms_of_glyphs_mswindows ();
1519       syms_of_gui_mswindows ();
1520 #ifdef HAVE_MENUBARS
1521       syms_of_menubar_mswindows ();
1522 #endif
1523 #ifdef HAVE_SCROLLBARS
1524       syms_of_scrollbar_mswindows ();
1525 #endif
1526 #endif  /* HAVE_MS_WINDOWS */
1527 #ifdef HAVE_MSW_C_DIRED
1528       syms_of_dired_mswindows ();
1529 #endif
1530 #ifdef WIN32_NATIVE
1531       syms_of_ntproc ();
1532 #endif
1533 #if defined (WIN32_NATIVE) || defined (CYGWIN)
1534       syms_of_win32 ();
1535 #endif
1536
1537 #ifdef MULE
1538       syms_of_mule ();
1539       syms_of_mule_ccl ();
1540       syms_of_mule_charset ();
1541 #endif
1542 #ifdef FILE_CODING
1543       syms_of_file_coding ();
1544 #endif
1545 #ifdef MULE
1546 #ifdef HAVE_WNN
1547       syms_of_mule_wnn ();
1548 #endif
1549 #ifdef HAVE_CANNA
1550       syms_of_mule_canna ();
1551 #endif /* HAVE_CANNA */
1552 #endif /* MULE */
1553
1554 #ifdef SYMS_SYSTEM
1555       SYMS_SYSTEM;
1556 #endif
1557
1558 #ifdef SYMS_MACHINE
1559       SYMS_MACHINE;
1560 #endif
1561
1562       /*
1563 #if defined (GNU_MALLOC) && \
1564     defined (ERROR_CHECK_MALLOC) && \
1565     !defined (HAVE_LIBMCHECK)
1566       */
1567       /* Prior to XEmacs 21, this was `#if 0'ed out. -slb */
1568 #if defined (LOSING_GCC_DESTRUCTOR_FREE_BUG)
1569       syms_of_free_hook ();
1570 #endif
1571
1572 #ifdef TOOLTALK
1573       syms_of_tooltalk ();
1574 #endif
1575
1576 #ifdef SUNPRO
1577       syms_of_sunpro ();
1578 #endif
1579
1580 #ifdef HAVE_LDAP
1581       syms_of_eldap ();
1582 #endif
1583
1584 #ifdef HAVE_GPM
1585       syms_of_gpmevent ();
1586 #endif
1587
1588 #ifdef HAVE_POSTGRESQL
1589       syms_of_postgresql ();
1590 #endif
1591
1592       /* Now create the subtypes for the types that have them.
1593          We do this before the vars_*() because more symbols
1594          may get initialized here. */
1595
1596       /* Now initialize the console types and associated symbols.
1597          Other than the first function below, the functions may
1598          make exactly the following function/macro calls:
1599
1600          INITIALIZE_CONSOLE_TYPE()
1601          CONSOLE_HAS_METHOD()
1602
1603          For any given console type, the former macro must be called
1604          before the any calls to the latter macro. */
1605
1606       console_type_create ();
1607
1608       console_type_create_stream ();
1609
1610 #ifdef HAVE_TTY
1611       console_type_create_tty ();
1612       console_type_create_device_tty ();
1613       console_type_create_frame_tty ();
1614       console_type_create_objects_tty ();
1615       console_type_create_redisplay_tty ();
1616 #endif
1617
1618 #ifdef HAVE_GTK
1619       console_type_create_gtk ();
1620       console_type_create_select_gtk ();
1621       console_type_create_device_gtk ();
1622       console_type_create_frame_gtk ();
1623       console_type_create_objects_gtk ();
1624       console_type_create_glyphs_gtk ();
1625       console_type_create_redisplay_gtk ();
1626 #ifdef HAVE_MENUBARS
1627       console_type_create_menubar_gtk ();
1628 #endif
1629 #ifdef HAVE_SCROLLBARS
1630       console_type_create_scrollbar_gtk ();
1631 #endif
1632 #ifdef HAVE_TOOLBARS
1633       console_type_create_toolbar_gtk ();
1634 #endif
1635 #ifdef HAVE_DIALOGS
1636       console_type_create_dialog_gtk ();
1637 #endif
1638 #endif /* HAVE_GTK */
1639
1640 #ifdef HAVE_X_WINDOWS
1641       console_type_create_x ();
1642       console_type_create_device_x ();
1643       console_type_create_frame_x ();
1644       console_type_create_glyphs_x ();
1645       console_type_create_select_x ();
1646 #ifdef HAVE_MENUBARS
1647       console_type_create_menubar_x ();
1648 #endif
1649       console_type_create_objects_x ();
1650       console_type_create_redisplay_x ();
1651 #ifdef HAVE_SCROLLBARS
1652       console_type_create_scrollbar_x ();
1653 #endif
1654 #ifdef HAVE_TOOLBARS
1655       console_type_create_toolbar_x ();
1656 #endif
1657 #ifdef HAVE_DIALOGS
1658       console_type_create_dialog_x ();
1659 #endif
1660 #endif /* HAVE_X_WINDOWS */
1661
1662 #ifdef HAVE_MS_WINDOWS
1663       console_type_create_mswindows ();
1664       console_type_create_device_mswindows ();
1665       console_type_create_frame_mswindows ();
1666       console_type_create_objects_mswindows ();
1667       console_type_create_redisplay_mswindows ();
1668       console_type_create_glyphs_mswindows ();
1669       console_type_create_select_mswindows ();
1670 # ifdef HAVE_SCROLLBARS
1671       console_type_create_scrollbar_mswindows ();
1672 # endif
1673 #ifdef HAVE_MENUBARS
1674       console_type_create_menubar_mswindows ();
1675 #endif
1676 #ifdef HAVE_TOOLBARS
1677       console_type_create_toolbar_mswindows ();
1678 #endif
1679 #ifdef HAVE_DIALOGS
1680       console_type_create_dialog_mswindows ();
1681 #endif
1682 #endif
1683
1684       /* Now initialize the specifier types and associated symbols.
1685          Other than the first function below, the functions may
1686          make exactly the following function/macro calls:
1687
1688          INITIALIZE_SPECIFIER_TYPE()
1689          SPECIFIER_HAS_METHOD()
1690
1691          For any given specifier type, the former macro must be called
1692          before the any calls to the latter macro. */
1693
1694       specifier_type_create ();
1695
1696       specifier_type_create_image ();
1697       specifier_type_create_gutter ();
1698       specifier_type_create_objects ();
1699 #ifdef HAVE_TOOLBARS
1700       specifier_type_create_toolbar ();
1701 #endif
1702
1703       /* Now initialize the structure types and associated symbols.
1704          Other than the first function below, the functions may
1705          make exactly the following function/macro calls:
1706
1707          define_structure_type()
1708          define_structure_type_keyword()
1709
1710          */
1711
1712       structure_type_create ();
1713
1714       structure_type_create_chartab ();
1715       structure_type_create_faces ();
1716       structure_type_create_rangetab ();
1717       structure_type_create_hash_table ();
1718
1719       /* Now initialize the image instantiator formats and associated symbols.
1720          Other than the first function below, the functions may
1721          make exactly the following function/macro calls:
1722
1723          INITIALIZE_IMAGE_INSTANTIATOR_FORMAT()
1724          IIFORMAT_HAS_METHOD()
1725          IIFORMAT_VALID_KEYWORD()
1726
1727          For any given image instantiator format, the first macro must be
1728          called before the any calls to the other macros. */
1729
1730       image_instantiator_format_create ();
1731       image_instantiator_format_create_glyphs_eimage ();
1732       image_instantiator_format_create_glyphs_widget ();
1733 #ifdef HAVE_TTY
1734       image_instantiator_format_create_glyphs_tty ();
1735 #endif
1736 #ifdef HAVE_X_WINDOWS
1737       image_instantiator_format_create_glyphs_x ();
1738 #endif /* HAVE_X_WINDOWS */
1739 #ifdef HAVE_MS_WINDOWS
1740       image_instantiator_format_create_glyphs_mswindows ();
1741 #endif /* HAVE_MSWINDOWS_WINDOWS */
1742 #ifdef HAVE_GTK
1743       image_instantiator_format_create_glyphs_gtk ();
1744 #endif
1745
1746       /* Now initialize the lstream types and associated symbols.
1747          Other than the first function below, the functions may
1748          make exactly the following function/macro calls:
1749
1750          LSTREAM_HAS_METHOD()
1751
1752          */
1753
1754       lstream_type_create ();
1755 #ifdef FILE_CODING
1756       lstream_type_create_file_coding ();
1757 #endif
1758 #if defined (HAVE_MS_WINDOWS) && !defined(HAVE_MSG_SELECT)
1759       lstream_type_create_mswindows_selectable ();
1760 #endif
1761
1762       /* Initialize processes implementation.
1763          The functions may make exactly the following function/macro calls:
1764
1765          PROCESS_HAS_METHOD()
1766       */
1767 #ifdef HAVE_UNIX_PROCESSES
1768       process_type_create_unix ();
1769 #endif
1770 #ifdef HAVE_WIN32_PROCESSES
1771       process_type_create_nt ();
1772 #endif
1773
1774       /* Now initialize most variables.
1775
1776          These functions may do exactly the following:
1777
1778          DEFVAR_INT()
1779          DEFVAR_LISP()
1780          DEFVAR_BOOL()
1781          DEFER_GETTEXT()
1782          Dynarr_*()
1783          Blocktype_*()
1784          staticpro()
1785          Fprovide(symbol)
1786          intern()
1787          Fput()
1788          xmalloc()
1789          defsymbol(), if it's absolutely necessary and you're sure that
1790            the symbol isn't referenced anywhere else in the initialization
1791            code
1792          Fset() on a symbol that is unbound
1793          assigning a symbol or constant value to a variable
1794          using a global variable that has been initialized
1795            earlier on in the same function
1796
1797          Any of the object-creating functions in alloc.c: e.g.
1798
1799          make_pure_*()
1800          make_string()
1801          build_string()
1802          make_vector()
1803          make_int()
1804          make_extent()
1805          alloc_lcrecord()
1806          Fcons()
1807          listN()
1808          make_opaque_ptr()
1809
1810          perhaps a few others.
1811
1812          NB:  Initialization or assignment should not be done here to certain
1813            variables settable from the command line.  See the comment above
1814            the call to pdump_load() in main_1().  This caveat should only
1815            apply to vars_of_emacs().
1816        */
1817
1818       /* Now allow Fprovide() statements to be made. */
1819       init_provide_once ();
1820
1821       /* Do that before any specifier creation (esp. vars_of_glyphs()) */
1822       vars_of_specifier ();
1823
1824       vars_of_abbrev ();
1825       vars_of_alloc ();
1826       vars_of_buffer ();
1827       vars_of_bytecode ();
1828       vars_of_callint ();
1829       vars_of_callproc ();
1830       vars_of_chartab ();
1831       vars_of_cmdloop ();
1832       vars_of_cmds ();
1833       vars_of_console ();
1834       vars_of_data ();
1835 #ifdef DEBUG_XEMACS
1836       vars_of_debug ();
1837       vars_of_tests ();
1838 #endif
1839       vars_of_console_stream ();
1840       vars_of_device ();
1841 #ifdef HAVE_DIALOGS
1842       vars_of_dialog ();
1843 #endif
1844       vars_of_dired ();
1845       vars_of_doc ();
1846 #ifdef HAVE_DRAGNDROP
1847       vars_of_dragdrop ();
1848 #endif
1849       vars_of_editfns ();
1850       vars_of_elhash ();
1851       vars_of_emacs ();
1852       vars_of_eval ();
1853
1854 #ifdef HAVE_X_WINDOWS
1855       vars_of_event_Xt ();
1856 #endif
1857 #if defined(HAVE_TTY) && (defined (DEBUG_TTY_EVENT_STREAM) || !defined (HAVE_X_WINDOWS))
1858       vars_of_event_tty ();
1859 #endif
1860 #ifdef HAVE_MS_WINDOWS
1861       vars_of_event_mswindows ();
1862 #endif
1863       vars_of_event_stream ();
1864
1865       vars_of_events ();
1866       vars_of_extents ();
1867       vars_of_faces ();
1868       vars_of_fileio ();
1869 #ifdef CLASH_DETECTION
1870       vars_of_filelock ();
1871 #endif
1872       vars_of_floatfns ();
1873       vars_of_font_lock ();
1874       vars_of_frame ();
1875       vars_of_glyphs ();
1876       vars_of_glyphs_eimage ();
1877       vars_of_glyphs_widget ();
1878       vars_of_gui ();
1879       vars_of_gutter ();
1880       vars_of_indent ();
1881       vars_of_insdel ();
1882       vars_of_intl ();
1883 #ifdef HAVE_XIM
1884 #ifdef XIM_MOTIF
1885       vars_of_input_method_motif ();
1886 #else /* XIM_XLIB */
1887       vars_of_input_method_xlib ();
1888 #endif
1889 #endif /* HAVE_XIM */
1890       vars_of_keymap ();
1891       vars_of_lread ();
1892       vars_of_lstream ();
1893       vars_of_macros ();
1894       vars_of_md5 ();
1895 #ifdef HAVE_DATABASE
1896       vars_of_database ();
1897 #endif
1898 #ifdef HAVE_MENUBARS
1899       vars_of_menubar ();
1900 #endif
1901       vars_of_minibuf ();
1902 #ifdef HAVE_SHLIB
1903       vars_of_module ();
1904 #endif
1905 #ifdef WIN32_NATIVE
1906       vars_of_nt ();
1907       vars_of_ntproc ();
1908 #endif
1909       vars_of_objects ();
1910       vars_of_print ();
1911
1912 #ifndef NO_SUBPROCESSES
1913       vars_of_process ();
1914 #ifdef HAVE_UNIX_PROCESSES
1915       vars_of_process_unix ();
1916 #endif
1917 #ifdef HAVE_WIN32_PROCESSES
1918       vars_of_process_nt ();
1919 #endif
1920 #endif
1921
1922       vars_of_profile ();
1923 #if defined (HAVE_MMAP) && defined (REL_ALLOC) && !defined(DOUG_LEA_MALLOC)
1924       vars_of_ralloc ();
1925 #endif /* HAVE_MMAP && REL_ALLOC */
1926       vars_of_redisplay ();
1927 #ifdef HAVE_SCROLLBARS
1928       vars_of_scrollbar ();
1929 #endif
1930       vars_of_search ();
1931       vars_of_select ();
1932       vars_of_sound ();
1933       vars_of_symbols ();
1934       vars_of_syntax ();
1935 #ifdef HAVE_TOOLBARS
1936       vars_of_toolbar ();
1937 #endif
1938       vars_of_undo ();
1939       vars_of_window ();
1940
1941 #ifdef HAVE_TTY
1942       vars_of_console_tty ();
1943       vars_of_frame_tty ();
1944       vars_of_objects_tty ();
1945 #endif
1946
1947 #ifdef HAVE_GTK
1948       vars_of_device_gtk ();
1949 #ifdef HAVE_DIALOGS
1950       vars_of_dialog_gtk ();
1951 #endif
1952       vars_of_event_gtk ();
1953       vars_of_frame_gtk ();
1954       vars_of_glyphs_gtk ();
1955       vars_of_ui_gtk ();
1956 #ifdef HAVE_MENUBARS
1957       vars_of_menubar_gtk ();
1958 #endif
1959       vars_of_objects_gtk ();
1960       vars_of_select_gtk ();
1961 #ifdef HAVE_SCROLLBARS
1962       vars_of_scrollbar_gtk ();
1963 #endif
1964 #if defined (HAVE_MENUBARS) || defined (HAVE_SCROLLBARS) || defined (HAVE_DIALOGS) || defined (HAVE_TOOLBARS)
1965       vars_of_gui_gtk ();
1966 #endif
1967 #endif /* HAVE_GTK */
1968
1969 #ifdef HAVE_X_WINDOWS
1970 #ifdef HAVE_BALLOON_HELP
1971       vars_of_balloon_x ();
1972 #endif
1973       vars_of_device_x ();
1974 #ifdef HAVE_DIALOGS
1975       vars_of_dialog_x ();
1976 #endif
1977       vars_of_frame_x ();
1978       vars_of_glyphs_x ();
1979 #ifdef HAVE_MENUBARS
1980       vars_of_menubar_x ();
1981 #endif
1982       vars_of_objects_x ();
1983       vars_of_select_x ();
1984 #ifdef HAVE_SCROLLBARS
1985       vars_of_scrollbar_x ();
1986 #endif
1987 #if defined (HAVE_MENUBARS) || defined (HAVE_SCROLLBARS) || defined (HAVE_DIALOGS) || defined (HAVE_TOOLBARS)
1988       vars_of_gui_x ();
1989 #endif
1990 #endif /* HAVE_X_WINDOWS */
1991
1992
1993 #ifdef HAVE_MS_WINDOWS
1994       vars_of_device_mswindows ();
1995       vars_of_console_mswindows ();
1996       vars_of_frame_mswindows ();
1997       vars_of_objects_mswindows ();
1998       vars_of_select_mswindows ();
1999       vars_of_glyphs_mswindows ();
2000 #ifdef HAVE_SCROLLBARS
2001       vars_of_scrollbar_mswindows ();
2002 #endif
2003 #ifdef HAVE_MENUBARS
2004       vars_of_menubar_mswindows ();
2005 #endif
2006 #ifdef HAVE_MSW_C_DIRED
2007       vars_of_dired_mswindows ();
2008 #endif
2009 #ifdef HAVE_DIALOGS
2010       vars_of_dialog_mswindows ();
2011 #endif
2012 #endif  /* HAVE_MS_WINDOWS */
2013
2014 #ifdef MULE
2015       vars_of_mule ();
2016       vars_of_mule_ccl ();
2017       vars_of_mule_charset ();
2018 #endif
2019 #ifdef FILE_CODING
2020       vars_of_file_coding ();
2021 #endif
2022 #ifdef MULE
2023 #ifdef HAVE_WNN
2024       vars_of_mule_wnn ();
2025 #endif
2026 #ifdef HAVE_CANNA
2027       vars_of_mule_canna ();
2028 #endif /* HAVE_CANNA */
2029 #endif /* MULE */
2030
2031 #ifdef TOOLTALK
2032       vars_of_tooltalk ();
2033 #endif
2034
2035 #ifdef SUNPRO
2036       vars_of_sunpro ();
2037 #endif
2038
2039 #ifdef HAVE_LDAP
2040       vars_of_eldap ();
2041 #endif
2042
2043 #ifdef HAVE_POSTGRESQL
2044       vars_of_postgresql();
2045 #endif
2046
2047 #ifdef HAVE_GPM
2048       vars_of_gpmevent ();
2049 #endif
2050
2051       /* Now initialize any specifier variables.  We do this later
2052          because it has some dependence on the vars initialized
2053          above.
2054
2055          These functions should *only* initialize specifier variables,
2056          and may make use of the following functions/macros in addition
2057          to the ones listed above:
2058
2059          DEFVAR_SPECIFIER()
2060          Fmake_specifier()
2061          set_specifier_fallback()
2062          set_specifier_caching()
2063          */
2064
2065       specifier_vars_of_glyphs ();
2066       specifier_vars_of_glyphs_widget ();
2067       specifier_vars_of_gutter ();
2068 #ifdef HAVE_MENUBARS
2069       specifier_vars_of_menubar ();
2070 #endif
2071       specifier_vars_of_redisplay ();
2072 #ifdef HAVE_SCROLLBARS
2073       specifier_vars_of_scrollbar ();
2074 #endif
2075 #ifdef HAVE_TOOLBARS
2076       specifier_vars_of_toolbar ();
2077 #endif
2078       specifier_vars_of_window ();
2079
2080       /* Now comes all the rest of the variables that couldn't
2081          be handled above.  There may be dependencies on variables
2082          initialized above, and dependencies between one complex_vars_()
2083          function and another. */
2084
2085       /* Calls Fmake_range_table(). */
2086       complex_vars_of_regex ();
2087       /* Calls Fmake_range_table(). */
2088       complex_vars_of_search ();
2089
2090       /* Calls make_lisp_hash_table(). */
2091       complex_vars_of_extents ();
2092
2093       /* Depends on hash tables and specifiers. */
2094       complex_vars_of_faces ();
2095
2096 #ifdef MULE
2097       /* These two depend on hash tables and various variables declared
2098          earlier.  The second may also depend on the first. */
2099       complex_vars_of_mule_charset ();
2100 #endif
2101 #ifdef FILE_CODING
2102       complex_vars_of_file_coding ();
2103 #endif
2104
2105       /* This calls allocate_glyph(), which creates specifiers
2106          and also relies on a variable (Vthe_nothing_vector) initialized
2107          above.  It also calls make_ext_string(), which under Mule
2108          could require that the charsets be initialized. */
2109       complex_vars_of_glyphs ();
2110
2111       /* These rely on the glyphs just created in the previous function,
2112          and call Fadd_spec_to_specifier(), which relies on various
2113          variables initialized above. */
2114 #ifdef HAVE_GTK
2115       complex_vars_of_glyphs_gtk ();
2116 #endif
2117 #ifdef HAVE_X_WINDOWS
2118       complex_vars_of_glyphs_x ();
2119 #endif
2120 #ifdef HAVE_MS_WINDOWS
2121       complex_vars_of_glyphs_mswindows ();
2122 #endif
2123
2124       /* This calls Fmake_glyph_internal(). */
2125       complex_vars_of_alloc ();
2126
2127       /* This calls Fmake_glyph_internal(). */
2128 #ifdef HAVE_MENUBARS
2129       complex_vars_of_menubar ();
2130 #endif
2131
2132       /* This calls Fmake_glyph_internal(). */
2133 #ifdef HAVE_SCROLLBARS
2134       complex_vars_of_scrollbar ();
2135 #endif
2136
2137       /* This calls allocate_glyph(). */
2138       complex_vars_of_frame ();
2139
2140       /* This calls Fcopy_category_table() under Mule, which calls who
2141          knows what. */
2142       complex_vars_of_chartab ();
2143
2144       /* This calls set_string_char(), which (under Mule) depends on the
2145          charsets being initialized. */
2146       complex_vars_of_casetab ();
2147
2148       /* This calls Fcopy_syntax_table(), which relies on char tables. */
2149       complex_vars_of_syntax ();
2150
2151       /* This initializes buffer-local variables, sets things up so
2152          that buffers can be created, and creates a couple of basic
2153          buffers.  This depends on Vstandard_syntax_table and
2154          Vstandard_category_table (initialized in the previous
2155          functions), as well as a whole horde of variables that may
2156          have been initialized above. */
2157       complex_vars_of_buffer ();
2158
2159       /* This initializes console-local variables. */
2160       complex_vars_of_console ();
2161
2162       /* This creates a couple more buffers, and depends on the
2163          previous function. */
2164       complex_vars_of_minibuf ();
2165
2166       /* These two might call Ffile_name_as_directory(), which
2167          might depend on all sorts of things; I'm not sure. */
2168       complex_vars_of_emacs ();
2169
2170       /* This creates a couple of basic keymaps and depends on Lisp
2171          hash tables and Ffset() (both of which depend on some variables
2172          initialized in the vars_of_*() section) and possibly other
2173          stuff. */
2174       complex_vars_of_keymap ();
2175
2176       /* Calls make_lisp_hash_table() and creates a keymap */
2177       complex_vars_of_event_stream ();
2178
2179 #ifdef ERROR_CHECK_GC
2180       {
2181         extern int always_gc;
2182         if (always_gc)                /* purification debugging hack */
2183           garbage_collect_1 ();
2184       }
2185 #endif
2186 #ifdef PDUMP
2187     } else if (!restart) {            /* after successful pdump_load() */
2188       reinit_alloc_once_early ();
2189       reinit_symbols_once_early ();
2190       reinit_opaque_once_early ();
2191
2192       reinit_console_type_create_stream ();
2193 #ifdef HAVE_TTY
2194       reinit_console_type_create_tty ();
2195 #endif
2196 #ifdef HAVE_X_WINDOWS
2197       reinit_console_type_create_x ();
2198       reinit_console_type_create_device_x ();
2199 #endif
2200 #ifdef HAVE_MS_WINDOWS
2201       reinit_console_type_create_mswindows ();
2202 #endif
2203 #ifdef HAVE_GTK
2204       reinit_console_type_create_gtk ();
2205 #endif
2206
2207       reinit_specifier_type_create ();
2208       reinit_specifier_type_create_image ();
2209       reinit_specifier_type_create_gutter ();
2210       reinit_specifier_type_create_objects ();
2211 #ifdef HAVE_TOOLBARS
2212       reinit_specifier_type_create_toolbar ();
2213 #endif
2214
2215       structure_type_create ();
2216
2217       structure_type_create_chartab ();
2218       structure_type_create_faces ();
2219       structure_type_create_rangetab ();
2220       structure_type_create_hash_table ();
2221
2222       lstream_type_create ();
2223 #ifdef FILE_CODING
2224       lstream_type_create_file_coding ();
2225 #endif
2226 #if defined (HAVE_MS_WINDOWS) && !defined(HAVE_MSG_SELECT)
2227       lstream_type_create_mswindows_selectable ();
2228 #endif
2229 #ifdef HAVE_UNIX_PROCESSES
2230       process_type_create_unix ();
2231 #endif
2232 #ifdef HAVE_WIN32_PROCESSES
2233       process_type_create_nt ();
2234 #endif
2235
2236       reinit_vars_of_buffer ();
2237       reinit_vars_of_console ();
2238 #ifdef DEBUG_XEMACS
2239       reinit_vars_of_debug ();
2240 #endif
2241       reinit_vars_of_device ();
2242       reinit_vars_of_eval ();
2243 #ifdef HAVE_X_WINDOWS
2244       reinit_vars_of_event_Xt ();
2245 #endif
2246 #ifdef HAVE_GTK
2247       reinit_vars_of_event_gtk ();
2248 #endif
2249 #if defined(HAVE_TTY) && (defined (DEBUG_TTY_EVENT_STREAM) || !defined (HAVE_X_WINDOWS))
2250       reinit_vars_of_event_tty ();
2251 #endif
2252 #ifdef HAVE_MS_WINDOWS
2253       reinit_vars_of_event_mswindows ();
2254 #endif
2255       reinit_vars_of_event_stream ();
2256       reinit_vars_of_events ();
2257       reinit_vars_of_extents ();
2258       reinit_vars_of_fileio ();
2259       reinit_vars_of_font_lock ();
2260       reinit_vars_of_glyphs ();
2261       reinit_vars_of_glyphs_widget ();
2262       reinit_vars_of_insdel ();
2263       reinit_vars_of_lread ();
2264       reinit_vars_of_lstream ();
2265       reinit_vars_of_minibuf ();
2266 #ifdef HAVE_SHLIB
2267       reinit_vars_of_module ();
2268 #endif
2269       reinit_vars_of_objects ();
2270       reinit_vars_of_print ();
2271       reinit_vars_of_search ();
2272       reinit_vars_of_undo ();
2273       reinit_vars_of_window ();
2274
2275 #ifdef HAVE_MS_WINDOWS
2276       reinit_vars_of_frame_mswindows ();
2277 #endif
2278
2279 #ifdef HAVE_GTK
2280       reinit_vars_of_menubar_gtk ();
2281 #endif
2282
2283 #ifdef HAVE_X_WINDOWS
2284       reinit_vars_of_device_x ();
2285 #ifdef HAVE_SCROLLBARS
2286       reinit_vars_of_scrollbar_x ();
2287 #endif
2288 #ifdef HAVE_MENUBARS
2289       reinit_vars_of_menubar_x ();
2290 #endif
2291       reinit_vars_of_select_x ();
2292 #if defined (HAVE_MENUBARS) || defined (HAVE_SCROLLBARS) || defined (HAVE_DIALOGS) || defined (HAVE_TOOLBARS)
2293       reinit_vars_of_gui_x ();
2294 #endif
2295 #endif /* HAVE_X_WINDOWS */
2296
2297 #if defined(MULE) && defined(HAVE_WNN)
2298       reinit_vars_of_mule_wnn ();
2299 #endif
2300
2301       reinit_complex_vars_of_buffer ();
2302       reinit_complex_vars_of_console ();
2303       reinit_complex_vars_of_minibuf ();
2304 #endif /* PDUMP */
2305     }
2306
2307
2308   /* CONGRATULATIONS!!!  We have successfully initialized the Lisp
2309      engine. */
2310
2311   if (initialized)
2312     {
2313       /* Stuff that should not be done at dump time, including stuff that
2314          needs to be reset at run time.  Order below should not matter.
2315
2316          Many initializations taken from the environment should go here. */
2317       reinit_alloc ();
2318       reinit_eval ();
2319 #ifdef MULE_REGEXP
2320       reinit_mule_category ();
2321 #endif
2322 #ifdef HAVE_POSTGRESQL
2323       init_postgresql_from_environment();
2324 #endif
2325     }
2326
2327   /* Now do further initialization/setup of stuff that is not needed by the
2328      syms_of_() routines.  This involves stuff that only is enabled in
2329      an interactive run (redisplay, user input, etc.) and stuff that is
2330      not needed until we start loading Lisp code (the reader).  A lot
2331      of this stuff involves querying the current environment and needs
2332      to be done both at dump time and at run time. */
2333
2334   init_initial_directory();             /* get the directory to use for the
2335                                            "*scratch*" buffer, etc. */
2336
2337 #ifdef WIN32_NATIVE
2338   /*
2339    * For Win32, call init_environment() now, so that environment/registry
2340    * variables will be properly entered into Vprocess_environment.
2341    */
2342   init_environment();
2343 #endif
2344
2345   init_callproc ();     /* Set up the process environment (so that egetenv
2346                            works), the basic directory variables
2347                            (exec-directory and so on), and stuff
2348                            related to subprocesses.  This should be
2349                            first because many of the functions below
2350                            call egetenv() to get environment variables. */
2351   init_lread ();        /* Set up the Lisp reader. */
2352   init_cmdargs (argc, (Extbyte **) argv,
2353                 skip_args);     /* Create list Vcommand_line_args */
2354   init_buffer ();       /* Set default directory of *scratch* buffer */
2355
2356 #ifdef WIN32_NATIVE
2357   init_ntproc();
2358 #endif
2359
2360   init_redisplay ();      /* Determine terminal type.
2361                              init_sys_modes uses results */
2362   init_frame ();
2363   init_event_stream (); /* Set up so we can get user input. */
2364   init_macros (); /* set up so we can run macros. */
2365   init_editfns (); /* Determine the name of the user we're running as */
2366   init_xemacs_process (); /* set up for calling subprocesses */
2367 #ifdef SUNPRO
2368   init_sunpro (); /* Set up Sunpro usage tracking */
2369 #endif
2370 #if defined (WIN32_NATIVE) || defined (CYGWIN)
2371   init_win32 ();
2372 #endif
2373 #if defined (HAVE_NATIVE_SOUND) && defined (hp9000s800)
2374   init_hpplay ();
2375 #endif
2376 #ifdef HAVE_TTY
2377   init_device_tty ();
2378 #endif
2379   init_console_stream (restart); /* Create the first console */
2380
2381   /* try to get the actual pathname of the exec file we are running */
2382   if (!restart)
2383   {
2384     Vinvocation_name = Fcar (Vcommand_line_args);
2385     if (XSTRING_DATA(Vinvocation_name)[0] == '-')
2386       {
2387         /* XEmacs as a login shell, oh goody! */
2388         Vinvocation_name = build_string(getenv("SHELL"));
2389       }
2390     Vinvocation_directory = Vinvocation_name;
2391
2392     if (!NILP (Ffile_name_directory (Vinvocation_name)))
2393       {
2394         /* invocation-name includes a directory component -- presumably it
2395            is relative to cwd, not $PATH */
2396         Vinvocation_directory = Fexpand_file_name (Vinvocation_name,
2397                                                    Qnil);
2398         Vinvocation_path = Qnil;
2399       }
2400     else
2401       {
2402         Vinvocation_path = decode_env_path ("PATH", NULL);
2403         locate_file (Vinvocation_path, Vinvocation_name,
2404                      Vlisp_EXEC_SUFFIXES,
2405                      &Vinvocation_directory, X_OK);
2406       }
2407
2408     if (NILP (Vinvocation_directory))
2409       Vinvocation_directory = Vinvocation_name;
2410
2411     Vinvocation_name = Ffile_name_nondirectory (Vinvocation_directory);
2412     Vinvocation_directory = Ffile_name_directory (Vinvocation_directory);
2413   }
2414
2415 #if defined(HAVE_SHLIB) && !defined(WIN32_NATIVE)
2416   /* This is Unix only.  MS Windows NT has a library call that does
2417      The Right Thing on that system.  Rumor has it, this must be
2418      called for GNU dld in temacs and xemacs.  */
2419   {
2420     char *buf = (char *)alloca (XSTRING_LENGTH (Vinvocation_directory)
2421                                 + XSTRING_LENGTH (Vinvocation_name)
2422                                 + 2);
2423     sprintf (buf, "%s/%s", XSTRING_DATA (Vinvocation_directory),
2424              XSTRING_DATA (Vinvocation_name));
2425
2426     /* All we can do is cry if an error happens, so ignore it. */
2427     (void) dll_init (buf);
2428   }
2429 #endif
2430
2431 #if defined (LOCALTIME_CACHE) && defined (HAVE_TZSET)
2432   /* sun's localtime() has a bug.  it caches the value of the time
2433      zone rather than looking it up every time.  Since localtime() is
2434      called to bolt the undumping time into the undumped emacs, this
2435      results in localtime() ignoring the TZ environment variable.
2436      This flushes the new TZ value into localtime(). */
2437   tzset ();
2438 #endif /* LOCALTIME_CACHE and TZSET */
2439
2440   load_me = Qnil;
2441   if (!initialized)
2442     {
2443       /* Handle -l loadup-and-dump, args passed by Makefile. */
2444       if (argc > 2 + skip_args && !strcmp (argv[1 + skip_args], "-l"))
2445         load_me = build_string (argv[2 + skip_args]);
2446 #if 0 /* CANNOT_DUMP - this can never be right in XEmacs --andyp */
2447       /* Unless next switch is -nl, load "loadup.el" first thing.  */
2448       if (!(argc > 1 + skip_args && !strcmp (argv[1 + skip_args], "-nl")))
2449         load_me = build_string ("loadup.el");
2450 #endif /* CANNOT_DUMP */
2451     }
2452
2453 #ifdef QUANTIFY
2454   if (initialized)
2455     quantify_start_recording_data ();
2456 #endif /* QUANTIFY */
2457
2458   initialized = 1;
2459   inhibit_non_essential_printing_operations = 0;
2460
2461   /* This never returns.  */
2462   initial_command_loop (load_me);
2463   /* NOTREACHED */
2464 }
2465
2466 \f
2467 /* Sort the args so we can find the most important ones
2468    at the beginning of argv.  */
2469
2470 /* First, here's a table of all the standard options.  */
2471
2472 struct standard_args
2473 {
2474   const char *name;
2475   const char *longname;
2476   int priority;
2477   int nargs;
2478 };
2479
2480 static const struct standard_args standard_args[] =
2481 {
2482   /* Handled by main_1 above: */
2483   { "-sd", "--show-dump-id", 105, 0 },
2484   { "-t", "--terminal", 100, 1 },
2485   { "-nd", "--no-dump-file", 95, 0 },
2486   { "-nw", "--no-windows", 90, 0 },
2487   { "-batch", "--batch", 85, 0 },
2488   { "-debug-paths", "--debug-paths", 82, 0 },
2489   { "-help", "--help", 80, 0 },
2490   { "-version", "--version", 75, 0 },
2491   { "-V", 0, 75, 0 },
2492   { "-d", "--display", 80, 1 },
2493   { "-display", 0, 80, 1 },
2494   { "-NXHost",  0, 79, 0 },
2495   { "-MachLaunch", 0, 79, 0},
2496
2497   /* Handled by command-line-early in startup.el: */
2498   { "-q", "--no-init-file", 50, 0 },
2499   { "-unmapped", 0, 50, 0 },
2500   { "-no-init-file", 0, 50, 0 },
2501   { "-vanilla", "--vanilla", 50, 0 },
2502   { "-no-autoloads", "--no-autoloads", 50, 0 },
2503   { "-no-site-file", "--no-site-file", 40, 0 },
2504   { "-no-early-packages", "--no-early-packages", 35, 0 },
2505   { "-u", "--user", 30, 1 },
2506   { "-user", 0, 30, 1 },
2507   { "-debug-init", "--debug-init", 20, 0 },
2508   { "-debug-paths", "--debug-paths", 20, 0 },
2509
2510   /* Xt options: */
2511   { "-i", "--icon-type", 15, 0 },
2512   { "-itype", 0, 15, 0 },
2513   { "-iconic", "--iconic", 15, 0 },
2514   { "-bg", "--background-color", 10, 1 },
2515   { "-background", 0, 10, 1 },
2516   { "-fg", "--foreground-color", 10, 1 },
2517   { "-foreground", 0, 10, 1 },
2518   { "-bd", "--border-color", 10, 1 },
2519   { "-bw", "--border-width", 10, 1 },
2520   { "-ib", "--internal-border", 10, 1 },
2521   { "-ms", "--mouse-color", 10, 1 },
2522   { "-cr", "--cursor-color", 10, 1 },
2523   { "-fn", "--font", 10, 1 },
2524   { "-font", 0, 10, 1 },
2525   { "-g", "--geometry", 10, 1 },
2526   { "-geometry", 0, 10, 1 },
2527   { "-T", "--title", 10, 1 },
2528   { "-title", 0, 10, 1 },
2529   { "-name", "--name", 10, 1 },
2530   { "-xrm", "--xrm", 10, 1 },
2531   { "-r", "--reverse-video", 5, 0 },
2532   { "-rv", 0, 5, 0 },
2533   { "-reverse", 0, 5, 0 },
2534   { "-hb", "--horizontal-scroll-bars", 5, 0 },
2535   { "-vb", "--vertical-scroll-bars", 5, 0 },
2536
2537   /* These have the same priority as ordinary file name args,
2538      so they are not reordered with respect to those.  */
2539   { "-L", "--directory", 0, 1 },
2540   { "-directory", 0, 0, 1 },
2541   { "-l", "--load", 0, 1 },
2542   { "-load", 0, 0, 1 },
2543   { "-f", "--funcall", 0, 1 },
2544   { "-funcall", 0, 0, 1 },
2545   { "-eval", "--eval", 0, 1 },
2546   { "-insert", "--insert", 0, 1 },
2547   /* This should be processed after ordinary file name args and the like.  */
2548   { "-kill", "--kill", -10, 0 },
2549 };
2550
2551 /* Reorder the elements of ARGV (assumed to have ARGC elements)
2552    so that the highest priority ones come first.
2553    Do not change the order of elements of equal priority.
2554    If an option takes an argument, keep it and its argument together.  */
2555
2556 static void
2557 sort_args (int argc, char **argv)
2558 {
2559   char **new_argv = xnew_array (char *, argc);
2560   /* For each element of argv,
2561      the corresponding element of options is:
2562      0 for an option that takes no arguments,
2563      1 for an option that takes one argument, etc.
2564      -1 for an ordinary non-option argument.  */
2565   int *options  = xnew_array (int, argc);
2566   int *priority = xnew_array (int, argc);
2567   int to = 1;
2568   int from;
2569   int i;
2570   int end_of_options_p = 0;
2571
2572   /* Categorize all the options,
2573      and figure out which argv elts are option arguments.  */
2574   for (from = 1; from < argc; from++)
2575     {
2576       options[from] = -1;
2577       priority[from] = 0;
2578       /* Pseudo options "--" and "run-temacs" indicate end of options */
2579       if (!strcmp (argv[from], "--") ||
2580           !strcmp (argv[from], "run-temacs"))
2581         end_of_options_p = 1;
2582       if (!end_of_options_p && argv[from][0] == '-')
2583         {
2584           int match, thislen;
2585           char *equals;
2586
2587           /* Look for a match with a known old-fashioned option.  */
2588           for (i = 0; i < countof (standard_args); i++)
2589             if (!strcmp (argv[from], standard_args[i].name))
2590               {
2591                 options[from]  = standard_args[i].nargs;
2592                 priority[from] = standard_args[i].priority;
2593                 if (from + standard_args[i].nargs >= argc)
2594                   fatal ("Option `%s' requires an argument\n", argv[from]);
2595                 from += standard_args[i].nargs;
2596                 goto done;
2597               }
2598
2599           /* Look for a match with a known long option.
2600              MATCH is -1 if no match so far, -2 if two or more matches so far,
2601              >= 0 (the table index of the match) if just one match so far.  */
2602           if (argv[from][1] == '-')
2603             {
2604               match = -1;
2605               thislen = strlen (argv[from]);
2606               equals = strchr (argv[from], '=');
2607               if (equals != 0)
2608                 thislen = equals - argv[from];
2609
2610               for (i = 0; i < countof (standard_args); i++)
2611                 if (standard_args[i].longname
2612                     && !strncmp (argv[from], standard_args[i].longname,
2613                                  thislen))
2614                   {
2615                     if (match == -1)
2616                       match = i;
2617                     else
2618                       match = -2;
2619                   }
2620
2621               /* If we found exactly one match, use that.  */
2622               if (match >= 0)
2623                 {
2624                   options[from]  = standard_args[match].nargs;
2625                   priority[from] = standard_args[match].priority;
2626                   /* If --OPTION=VALUE syntax is used,
2627                      this option uses just one argv element.  */
2628                   if (equals != 0)
2629                     options[from] = 0;
2630                   if (from + options[from] >= argc)
2631                     fatal ("Option `%s' requires an argument\n", argv[from]);
2632                   from += options[from];
2633                 }
2634             }
2635         done: ;
2636         }
2637     }
2638
2639   /* Copy the arguments, in order of decreasing priority, to NEW_ARGV.  */
2640   new_argv[0] = argv[0];
2641   while (to < argc)
2642     {
2643       int best = -1;
2644       int best_priority = -9999;
2645
2646       /* Find the highest priority remaining option.
2647          If several have equal priority, take the first of them.  */
2648       for (from = 1; from < argc; from++)
2649         {
2650           if (argv[from] != 0 && priority[from] > best_priority)
2651             {
2652               best_priority = priority[from];
2653               best = from;
2654             }
2655           /* Skip option arguments--they are tied to the options.  */
2656           if (options[from] > 0)
2657             from += options[from];
2658         }
2659
2660       if (best < 0)
2661         abort ();
2662
2663       /* Copy the highest priority remaining option, with its args, to NEW_ARGV.  */
2664       new_argv[to++] = argv[best];
2665       for (i = 0; i < options[best]; i++)
2666         new_argv[to++] = argv[best + i + 1];
2667
2668       /* Clear out this option in ARGV.  */
2669       argv[best] = 0;
2670       for (i = 0; i < options[best]; i++)
2671         argv[best + i + 1] = 0;
2672     }
2673
2674   memcpy (argv, new_argv, sizeof (char *) * argc);
2675   xfree (new_argv);
2676   xfree (options);
2677   xfree (priority);
2678 }
2679
2680 DEFUN ("running-temacs-p", Frunning_temacs_p, 0, 0, 0, /*
2681 True if running temacs.  This means we are in the dumping stage.
2682 This is false during normal execution of the `xemacs' program, and
2683 becomes false once `run-emacs-from-temacs' is run.
2684 */
2685        ())
2686 {
2687   return run_temacs_argc >= 0 ? Qt : Qnil;
2688 }
2689
2690 DEFUN ("run-emacs-from-temacs", Frun_emacs_from_temacs, 0, MANY, 0, /*
2691 Do not call this.  It will reinitialize your XEmacs.  You'll be sorry.
2692 */
2693 /* If this function is called from startup.el, it will be possible to run
2694    temacs as an editor using 'temacs -batch -l loadup.el run-temacs', instead
2695    of having to dump an emacs and then run that (when debugging emacs itself,
2696    this can be much faster)). [Actually, the speed difference isn't that
2697    much as long as your filesystem is local, and you don't end up with
2698    a dumped version in case you want to rerun it.  This function is most
2699    useful when used as part of the `make all-elc' command. --ben]
2700    This will "restart" emacs with the specified command-line arguments.
2701
2702    Martin thinks this function is most useful when using debugging
2703    tools like Purify or tcov that get confused by XEmacs' dumping.  */
2704      (int nargs, Lisp_Object *args))
2705 {
2706   int ac;
2707   const Extbyte *wampum;
2708   int namesize;
2709   int total_len;
2710   Lisp_Object orig_invoc_name = Fcar (Vcommand_line_args);
2711   const Extbyte **wampum_all = alloca_array (const Extbyte *, nargs);
2712   int *wampum_all_len  = alloca_array (int, nargs);
2713
2714   assert (!gc_in_progress);
2715
2716   if (run_temacs_argc < 0)
2717     error ("I've lost my temacs-hood.");
2718
2719   /* Need to convert the orig_invoc_name and all of the arguments
2720      to external format. */
2721
2722   TO_EXTERNAL_FORMAT (LISP_STRING, orig_invoc_name,
2723                       ALLOCA, (wampum, namesize),
2724                       Qnative);
2725   namesize++;
2726
2727   for (ac = 0, total_len = namesize; ac < nargs; ac++)
2728     {
2729       CHECK_STRING (args[ac]);
2730       TO_EXTERNAL_FORMAT (LISP_STRING, args[ac],
2731                           ALLOCA, (wampum_all[ac], wampum_all_len[ac]),
2732                           Qnative);
2733       wampum_all_len[ac]++;
2734       total_len += wampum_all_len[ac];
2735     }
2736   DO_REALLOC (run_temacs_args, run_temacs_args_size, total_len, char);
2737   DO_REALLOC (run_temacs_argv, run_temacs_argv_size, nargs+2, char *);
2738
2739   memcpy (run_temacs_args, wampum, namesize);
2740   run_temacs_argv [0] = run_temacs_args;
2741   for (ac = 0; ac < nargs; ac++)
2742     {
2743       memcpy (run_temacs_args + namesize,
2744               wampum_all[ac], wampum_all_len[ac]);
2745       run_temacs_argv [ac + 1] = run_temacs_args + namesize;
2746       namesize += wampum_all_len[ac];
2747     }
2748   run_temacs_argv [nargs + 1] = 0;
2749   catchlist = NULL; /* Important!  Otherwise free_cons() calls in
2750                        condition_case_unwind() may lead to GC death. */
2751   unbind_to (0, Qnil); /* this closes loadup.el */
2752   purify_flag = 0;
2753   run_temacs_argc = nargs + 1;
2754 #if defined (HEAP_IN_DATA) && !defined(PDUMP)
2755   report_sheap_usage (0);
2756 #endif
2757   LONGJMP (run_temacs_catch, 1);
2758   return Qnil; /* not reached; warning suppression */
2759 }
2760
2761 /* ARGSUSED */
2762 int
2763 main (int argc, char **argv, char **envp)
2764 {
2765
2766 #ifdef _MSC_VER
2767   /* Under VC++, access violations and the like are not sent through
2768      the standard signal() mechanism.  Rather, they need to be handled
2769      using the Microsoft "structured exception handling" mechanism,
2770      which vaguely resembles the C++ mechanisms. */
2771   __try
2772   {
2773 #endif
2774
2775   int     volatile vol_argc = argc;
2776   char ** volatile vol_argv = argv;
2777   char ** volatile vol_envp = envp;
2778   /* This is hairy.  We need to compute where the XEmacs binary was invoked
2779      from because temacs initialization requires it to find the lisp
2780      directories.  The code that recomputes the path is guarded by the
2781      restarted flag.  There are three possible paths I've found so far
2782      through this:
2783
2784      temacs -- When running temacs for basic build stuff, the first main_1
2785       will be the only one invoked.  It must compute the path else there
2786       will be a very ugly bomb in startup.el (can't find obvious location
2787       for doc-directory data-directory, etc.).
2788
2789      temacs w/ run-temacs on the command line -- This is run to bytecompile
2790       all the out of date dumped lisp.  It will execute both of the main_1
2791       calls and the second one must not touch the first computation because
2792       argc/argv are hosed the second time through.
2793
2794      xemacs -- Only the second main_1 is executed.  The invocation path must
2795       computed but this only matters when running in place or when running
2796       as a login shell.
2797
2798      As a bonus for straightening this out, XEmacs can now be run in place
2799      as a login shell.  This never used to work.
2800
2801      As another bonus, we can now guarantee that
2802      (concat invocation-directory invocation-name) contains the filename
2803      of the XEmacs binary we are running.  This can now be used in a
2804      definite test for out of date dumped files.  -slb */
2805   int restarted = 0;
2806 #ifdef QUANTIFY
2807   quantify_stop_recording_data ();
2808   quantify_clear_data ();
2809 #endif /* QUANTIFY */
2810
2811   inhibit_non_essential_printing_operations = 1;
2812   suppress_early_error_handler_backtrace = 0;
2813   lim_data = 0; /* force reinitialization of this variable */
2814
2815   /* Lisp_Object must fit in a word; check VALBITS and GCTYPEBITS */
2816   assert (sizeof (Lisp_Object) == sizeof (void *));
2817
2818 #ifdef LINUX_SBRK_BUG
2819   sbrk (1);
2820 #endif
2821
2822   if (!initialized)
2823     {
2824 #ifdef DOUG_LEA_MALLOC
2825       if (mallopt (M_MMAP_MAX, 0) != 1)
2826         abort();
2827 #endif
2828       run_temacs_argc = 0;
2829       if (! SETJMP (run_temacs_catch))
2830         {
2831           main_1 (vol_argc, vol_argv, vol_envp, 0);
2832         }
2833       /* run-emacs-from-temacs called */
2834       restarted = 1;
2835       vol_argc = run_temacs_argc;
2836       vol_argv = run_temacs_argv;
2837 #ifdef _SCO_DS
2838       /* This makes absolutely no sense to anyone involved.  There are
2839          several people using this stuff.  We've compared versions on
2840          everything we can think of.  We can find no difference.
2841          However, on both my systems environ is a plain old global
2842          variable initialized to zero.  _environ is the one that
2843          contains pointers to the actual environment.
2844
2845          Since we can't figure out the difference (and we're hours
2846          away from a release), this takes a very cowardly approach and
2847          is bracketed with both a system specific preprocessor test
2848          and a runtime "do you have this problem" test
2849
2850          06/20/96 robertl@dgii.com */
2851       {
2852         extern char **_environ;
2853         if ((unsigned) environ == 0)
2854           environ=_environ;
2855       }
2856 #endif /* _SCO_DS */
2857       vol_envp = environ;
2858     }
2859 #if defined (RUN_TIME_REMAP) && ! defined (PDUMP)
2860   else
2861     /* obviously no-one uses this because where it was before initialized was
2862      *always* true */
2863     run_time_remap (argv[0]);
2864 #endif
2865
2866 #ifdef DOUG_LEA_MALLOC
2867   if (initialized && (malloc_state_ptr != NULL))
2868     {
2869       int rc = malloc_set_state (malloc_state_ptr);
2870       if (rc != 0)
2871         {
2872           stderr_out ("malloc_set_state failed, rc = %d\n", rc);
2873           abort ();
2874         }
2875 #if 0
2876       free (malloc_state_ptr);
2877 #endif
2878       /* mmap works in glibc-2.1, glibc-2.0 (Non-Mule only) and Linux libc5 */
2879 #if (defined(__GLIBC__) && __GLIBC_MINOR__ >= 1) || \
2880     defined(_NO_MALLOC_WARNING_) || \
2881     (defined(__GLIBC__) && __GLIBC_MINOR__ < 1 && !defined(MULE)) || \
2882     defined(DEBUG_DOUG_LEA_MALLOC)
2883       if(mallopt (M_MMAP_MAX, 0) != 1)
2884         abort();
2885 #endif
2886 #ifdef REL_ALLOC
2887       r_alloc_reinit ();
2888 #endif
2889     }
2890 #endif /* DOUG_LEA_MALLOC */
2891
2892   run_temacs_argc = -1;
2893
2894   main_1 (vol_argc, vol_argv, vol_envp, restarted);
2895
2896 #ifdef _MSC_VER
2897   }
2898   /* VC++ documentation says that
2899      GetExceptionCode() cannot be called inside the filter itself. */
2900   __except (mswindows_handle_hardware_exceptions (GetExceptionCode ())) {}
2901 #endif
2902
2903   return 0; /* unreached */
2904 }
2905
2906 \f
2907 /* Dumping apparently isn't supported by versions of GCC >= 2.8. */
2908 /* The following needs conditionalization on whether either XEmacs or */
2909 /* various system shared libraries have been built and linked with */
2910 /* GCC >= 2.8.  -slb */
2911 #if defined(GNU_MALLOC)
2912 static void
2913 voodoo_free_hook (void *mem)
2914 {
2915   /* Disable all calls to free() when XEmacs is exiting and it doesn't */
2916   /* matter. */
2917   __free_hook =
2918 #if defined __GNUC__ || defined __INTEL_COMPILER
2919 /* prototype of __free_hook varies with glibc version */
2920     (__typeof__ (__free_hook))
2921 #endif
2922     voodoo_free_hook;
2923 }
2924 #endif /* GNU_MALLOC */
2925
2926 DEFUN ("kill-emacs", Fkill_emacs, 0, 1, "P", /*
2927 Exit the XEmacs job and kill it.  Ask for confirmation, without argument.
2928 If ARG is an integer, return ARG as the exit program code.
2929 If ARG is a string, stuff it as keyboard input.
2930
2931 The value of `kill-emacs-hook', if not void,
2932 is a list of functions (of no args),
2933 all of which are called before XEmacs is actually killed.
2934 */
2935        (arg))
2936 {
2937   /* This function can GC */
2938   struct gcpro gcpro1;
2939
2940   GCPRO1 (arg);
2941
2942   if (feof (stdin))
2943     arg = Qt;
2944
2945   if (!preparing_for_armageddon && !noninteractive)
2946     run_hook (Qkill_emacs_hook);
2947
2948   ensure_no_quitting_from_now_on ();
2949
2950   if (!preparing_for_armageddon)
2951     {
2952       Lisp_Object concons, nextcons;
2953
2954       /* Normally, go ahead and delete all the consoles now.
2955          Some unmentionably lame window systems (MS Wwwww...... eek,
2956          I can't even say it) don't properly clean up after themselves,
2957          and even for those that do, it might be cleaner this way.
2958          If we're going down, however, we don't do this (might
2959          be too dangerous), and if we get a crash somewhere within
2960          this loop, we'll still autosave and won't try this again. */
2961
2962       LIST_LOOP_DELETING (concons, nextcons, Vconsole_list)
2963         {
2964           /* There is very little point in deleting the stream console.
2965              It uses stdio, which should flush any buffered output and
2966              something can only go wrong. -slb */
2967           /* I changed my mind.  There's a stupid hack in close to add
2968              a trailing newline. */
2969           /*if (!CONSOLE_STREAM_P (XCONSOLE (XCAR (concons))))*/
2970             delete_console_internal (XCONSOLE (XCAR (concons)), 1, 1, 0);
2971         }
2972     }
2973
2974   UNGCPRO;
2975
2976 #ifdef HAVE_MS_WINDOWS
2977   /* If we displayed a message on the console, then we must allow the
2978      user to see this message.  This may be unnecessary, but can't
2979      hurt, and we can't necessarily check arg; e.g. xemacs --help
2980      kills with argument 0.
2981
2982      Don't do this in batch mode, it makes no sense and is more
2983      annoying than useful. --andyp */
2984   if (mswindows_message_outputted && !noninteractive)
2985     Fmswindows_message_box (build_string ("Messages outputted.  XEmacs is exiting."),
2986                             Qnil, Qnil);
2987 #endif
2988
2989   shut_down_emacs (0, STRINGP (arg) ? arg : Qnil, 0);
2990
2991 #if defined(GNU_MALLOC)
2992   __free_hook =
2993 #if defined __GNUC__ || defined __INTEL_COMPILER
2994 /* prototype of __free_hook varies with glibc version */
2995     (__typeof__ (__free_hook))
2996 #endif
2997     voodoo_free_hook;
2998 #endif
2999
3000   exit (INTP (arg) ? XINT (arg) : 0);
3001   /* NOTREACHED */
3002   return Qnil; /* I'm sick of the compiler warning */
3003 }
3004
3005 /* Perform an orderly shutdown of XEmacs.  Autosave any modified
3006    buffers, kill any child processes, clean up the terminal modes (if
3007    we're in the foreground), and other stuff like that.  Don't perform
3008    any redisplay; this may be called when XEmacs is shutting down in
3009    the background, or after its X connection has died.
3010
3011    If SIG is a signal number, print a message for it.
3012
3013    This is called by fatal signal handlers and Fkill_emacs.  It used to
3014    be called by X protocol error handlers, but instead they now call
3015    Fkill_emacs. */
3016 static void
3017 shut_down_emacs (int sig, Lisp_Object stuff, int no_auto_save)
3018 {
3019   /* This function can GC */
3020   /* Prevent running of hooks and other non-essential stuff
3021      from now on.  */
3022   preparing_for_armageddon = 1;
3023
3024   ensure_no_quitting_from_now_on ();
3025
3026 #ifdef QUANTIFY
3027   quantify_stop_recording_data ();
3028 #endif /* QUANTIFY */
3029
3030   /* This is absolutely the most important thing to do, so make sure
3031      we do it now, before anything else.  We might have crashed and
3032      be in a weird inconsistent state, and potentially anything could
3033      set off another protection fault and cause us to bail out
3034      immediately. */
3035   /* Steve writes the following:
3036
3037      [[I'm not removing the code entirely, yet.  We have run up against
3038      a spate of problems in diagnosing crashes due to crashes within
3039      crashes.  It has very definitely been determined that code called
3040      during auto-saving cannot work if XEmacs crashed inside of GC.
3041      We already auto-save on an itimer so there cannot be too much
3042      unsaved stuff around, and if we get better crash reports we might
3043      be able to get more problems fixed so I'm disabling this.  -slb]]
3044
3045      and DISABLES AUTO-SAVING ENTIRELY during crashes!  Way way bad idea.
3046
3047      Instead let's just be more intelligent about avoiding crashing
3048      when possible, esp. nested crashes.
3049   */
3050   if (!no_auto_save)
3051     Fdo_auto_save (Qt, Qnil); /* do this before anything hazardous */
3052
3053   fflush (stdout);
3054   reset_all_consoles ();
3055   if (sig && sig != SIGTERM)
3056     {
3057       if (sig == -1)
3058         stderr_out ("\nFatal error.\n\n");
3059       else
3060         stderr_out ("\nFatal error (%d).\n\n", sig);
3061       stderr_out
3062         ("Your files have been auto-saved.\n"
3063          "Use `M-x recover-session' to recover them.\n"
3064          "\n"
3065          "Your version of XEmacs was distributed with a PROBLEMS file that  may describe\n"
3066          "your crash, and with luck a workaround.  Please check it first, but do report\n"
3067          "the crash anyway.  "
3068 #ifdef INFODOCK
3069          "\n\nPlease report this bug by selecting `Report-Bug' in the InfoDock menu.\n"
3070          "*BE SURE* to include the XEmacs configuration from M-x describe-installation,\n"
3071          "or the file Installation in the top directory of the build tree.\n"
3072 #else
3073          "Please report this bug by invoking M-x report-emacs-bug,\n"
3074          "or by selecting `Send Bug Report' from the Help menu.  If necessary, send\n"
3075          "ordinary email to `crashes@xemacs.org'.  *MAKE SURE* to include the XEmacs\n"
3076          "configuration from M-x describe-installation, or equivalently the file\n"
3077          "Installation in the top of the build tree.\n"
3078 #endif
3079 #ifndef _MSC_VER
3080          "\n"
3081          "*Please* try *hard* to obtain a C stack backtrace; without it, we are unlikely\n"
3082          "to be able to analyze the problem.  Locate the core file produced as a result\n"
3083          "of this crash (often called `core' or `core.<process-id>', and located in\n"
3084          "the directory in which you started XEmacs or your home directory), and type\n"
3085          "\n"
3086          "  gdb "
3087 #endif
3088          );
3089 #ifndef _MSC_VER
3090       {
3091         const char *name;
3092         char *dir = 0;
3093
3094         /* Now try to determine the actual path to the executable,
3095            to try to make the backtrace-determination process as foolproof
3096            as possible. */
3097         if (STRINGP (Vinvocation_name))
3098           name = (char *) XSTRING_DATA (Vinvocation_name);
3099         else
3100           name = "xemacs";
3101         if (STRINGP (Vinvocation_directory))
3102           dir = (char *) XSTRING_DATA (Vinvocation_directory);
3103         if (!dir || dir[0] != '/')
3104           stderr_out ("`which %s`", name);
3105         else if (dir[strlen (dir) - 1] != '/')
3106           stderr_out ("%s/%s", dir, name);
3107         else
3108           stderr_out ("%s%s", dir, name);
3109       }
3110       stderr_out
3111         (" core\n"
3112          "\n"
3113          "then type `where' at the debugger prompt.  No GDB on your system?  You may\n"
3114          "have DBX, or XDB, or SDB.  (Ask your system administrator if you need help.)\n"
3115          "If no core file was produced, enable them (often with `ulimit -c unlimited'\n"
3116          "in case of future recurrance of the crash.\n");
3117 #endif /* _MSC_VER */
3118     }
3119
3120   stuff_buffered_input (stuff);
3121
3122   kill_buffer_processes (Qnil);
3123
3124 #ifdef CLASH_DETECTION
3125   unlock_all_files ();
3126 #endif
3127
3128 #ifdef TOOLTALK
3129   tt_session_quit (tt_default_session ());
3130 #if 0
3131   /* The following crashes when built on X11R5 and run on X11R6 */
3132   tt_close ();
3133 #endif
3134 #endif /* TOOLTALK */
3135
3136 }
3137
3138 \f
3139 #ifndef CANNOT_DUMP
3140
3141 #if !defined(PDUMP) || !defined(SYSTEM_MALLOC)
3142 extern char my_edata[];
3143 #endif
3144
3145 extern void disable_free_hook (void);
3146
3147 DEFUN ("dump-emacs", Fdump_emacs, 2, 2, 0, /*
3148 Dump current state of XEmacs into executable file FILENAME.
3149 Take symbols from SYMFILE (presumably the file you executed to run XEmacs).
3150 This is used in the file `loadup.el' when building XEmacs.
3151
3152 Remember to set `command-line-processed' to nil before dumping
3153 if you want the dumped XEmacs to process its command line
3154 and announce itself normally when it is run.
3155 */
3156        (filename, symfile))
3157 {
3158   /* This function can GC */
3159   struct gcpro gcpro1, gcpro2;
3160   int opurify;
3161
3162   GCPRO2 (filename, symfile);
3163
3164 #ifdef FREE_CHECKING
3165   Freally_free (Qnil);
3166
3167   /* When we're dumping, we can't use the debugging free() */
3168   disable_free_hook ();
3169 #endif
3170
3171   CHECK_STRING (filename);
3172   filename = Fexpand_file_name (filename, Qnil);
3173   if (!NILP (symfile))
3174     {
3175       CHECK_STRING (symfile);
3176       if (XSTRING_LENGTH (symfile) > 0)
3177         symfile = Fexpand_file_name (symfile, Qnil);
3178       else
3179         symfile = Qnil;
3180     }
3181
3182   opurify = purify_flag;
3183   purify_flag = 0;
3184
3185 #if defined (HEAP_IN_DATA) && !defined(PDUMP)
3186   report_sheap_usage (1);
3187 #endif
3188
3189   clear_message ();
3190
3191   fflush (stderr);
3192   fflush (stdout);
3193
3194   disksave_object_finalization ();
3195   release_breathing_space ();
3196
3197   /* Tell malloc where start of impure now is */
3198   /* Also arrange for warnings when nearly out of space.  */
3199 #ifndef SYSTEM_MALLOC
3200   memory_warnings (my_edata, malloc_warning);
3201 #endif
3202
3203   UNGCPRO;
3204
3205   {
3206     char *filename_ext;
3207     char *symfile_ext;
3208
3209     LISP_STRING_TO_EXTERNAL (filename, filename_ext, Qfile_name);
3210
3211     if (STRINGP (symfile))
3212       LISP_STRING_TO_EXTERNAL (symfile, symfile_ext, Qfile_name);
3213     else
3214       symfile_ext = 0;
3215
3216     garbage_collect_1 ();
3217
3218 #ifdef PDUMP
3219     pdump ();
3220 #else
3221
3222 #ifdef DOUG_LEA_MALLOC
3223     malloc_state_ptr = malloc_get_state ();
3224 #endif
3225   /* here we break our rule that the filename conversion should
3226      be performed at the actual time that the system call is made.
3227      It's a whole lot easier to do the conversion here than to
3228      modify all the unexec routines to ensure that filename
3229      conversion is applied everywhere.  Don't worry about memory
3230      leakage because this call only happens once. */
3231     unexec (filename_ext, symfile_ext, (uintptr_t) my_edata, 0, 0);
3232 #ifdef DOUG_LEA_MALLOC
3233     free (malloc_state_ptr);
3234 #endif
3235 #endif /* not PDUMP */
3236   }
3237
3238   purify_flag = opurify;
3239
3240   return Qnil;
3241 }
3242
3243 #endif /* not CANNOT_DUMP */
3244 \f
3245
3246 /* Split STRING into a list of substrings.  The substrings are the
3247    parts of original STRING separated by SEPCHAR.  */
3248 static Lisp_Object
3249 split_string_by_emchar_1 (const Bufbyte *string, Bytecount size,
3250                           Emchar sepchar)
3251 {
3252   Lisp_Object result = Qnil;
3253   const Bufbyte *end = string + size;
3254
3255   while (1)
3256     {
3257       const Bufbyte *p = string;
3258       while (p < end)
3259         {
3260           if (charptr_emchar (p) == sepchar)
3261             break;
3262           INC_CHARPTR (p);
3263         }
3264       result = Fcons (make_string (string, p - string), result);
3265       if (p < end)
3266         {
3267           string = p;
3268           INC_CHARPTR (string); /* skip sepchar */
3269         }
3270       else
3271         break;
3272     }
3273   return Fnreverse (result);
3274 }
3275
3276 /* The same as the above, except PATH is an external C string (it is
3277    converted using Qfile_name), and sepchar is hardcoded to SEPCHAR
3278    (':' or whatever).  */
3279 Lisp_Object
3280 decode_path (const char *path)
3281 {
3282   Bytecount newlen;
3283   Bufbyte *newpath;
3284   if (!path)
3285     return Qnil;
3286
3287   TO_INTERNAL_FORMAT (C_STRING, path, ALLOCA, (newpath, newlen), Qfile_name);
3288
3289   /* #### Does this make sense?  It certainly does for
3290      decode_env_path(), but it looks dubious here.  Does any code
3291      depend on decode_path("") returning nil instead of an empty
3292      string?  */
3293   if (!newlen)
3294     return Qnil;
3295
3296   return split_string_by_emchar_1 (newpath, newlen, SEPCHAR);
3297 }
3298
3299 Lisp_Object
3300 decode_env_path (const char *evarname, const char *default_)
3301 {
3302   const char *path = 0;
3303   if (evarname)
3304     path = egetenv (evarname);
3305   if (!path)
3306     path = default_;
3307   return decode_path (path);
3308 }
3309
3310 /* Ben thinks this function should not exist or be exported to Lisp.
3311    We use it to define split-path-string in subr.el (not!).  */
3312
3313 DEFUN ("split-string-by-char", Fsplit_string_by_char, 2, 2, 0, /*
3314 Split STRING into a list of substrings originally separated by SEPCHAR.
3315 */
3316        (string, sepchar))
3317 {
3318   CHECK_STRING (string);
3319   CHECK_CHAR (sepchar);
3320   return split_string_by_emchar_1 (XSTRING_DATA (string),
3321                                    XSTRING_LENGTH (string),
3322                                    XCHAR (sepchar));
3323 }
3324
3325 /* #### This was supposed to be in subr.el, but is used VERY early in
3326    the bootstrap process, so it goes here.  Damn.  */
3327
3328 DEFUN ("split-path", Fsplit_path, 1, 1, 0, /*
3329 Explode a search path into a list of strings.
3330 The path components are separated with the characters specified
3331 with `path-separator'.
3332 */
3333        (path))
3334 {
3335   CHECK_STRING (path);
3336
3337   while (!STRINGP (Vpath_separator)
3338          || (XSTRING_CHAR_LENGTH (Vpath_separator) != 1))
3339     Vpath_separator = signal_simple_continuable_error
3340       ("`path-separator' should be set to a single-character string",
3341        Vpath_separator);
3342
3343   return (split_string_by_emchar_1
3344           (XSTRING_DATA (path), XSTRING_LENGTH (path),
3345            charptr_emchar (XSTRING_DATA (Vpath_separator))));
3346 }
3347 \f
3348 DEFUN ("noninteractive", Fnoninteractive, 0, 0, 0, /*
3349 Non-nil return value means XEmacs is running without interactive terminal.
3350 */
3351        ())
3352 {
3353   return noninteractive ? Qt : Qnil;
3354 }
3355
3356 /* This flag is useful to define if you're under a debugger; this way, you
3357    can put a breakpoint of assert_failed() and debug multiple problems
3358    in one session without having to recompile. */
3359 /* #define ASSERTIONS_DONT_ABORT */
3360
3361 #ifdef USE_ASSERTIONS
3362 /* This highly dubious kludge ... shut up Jamie, I'm tired of your slagging. */
3363
3364 static int in_assert_failed;
3365 static const char *assert_failed_file;
3366 static int assert_failed_line;
3367 static const char *assert_failed_expr;
3368
3369 #ifdef fprintf
3370 #undef fprintf
3371 #endif
3372
3373 #undef abort    /* avoid infinite #define loop... */
3374
3375 #if defined (WIN32_NATIVE) && defined (DEBUG_XEMACS)
3376 #define enter_debugger() DebugBreak ()
3377 #else
3378 #define enter_debugger()
3379 #endif
3380
3381 void
3382 assert_failed (const char *file, int line, const char *expr)
3383 {
3384   /* If we're already crashing, let's not crash again.  This might be
3385      critical to getting auto-saving working properly. */
3386   if (fatal_error_in_progress)
3387     return;
3388
3389   /* We are extremely paranoid so we sensibly deal with recursive
3390      assertion failures. */
3391   in_assert_failed++;
3392   inhibit_non_essential_printing_operations = 1;
3393
3394   if (in_assert_failed >= 4)
3395     _exit (-1);
3396   else if (in_assert_failed == 3)
3397     {
3398       enter_debugger ();
3399       _exit (-1);
3400     }
3401   else if (in_assert_failed == 2)
3402     {
3403       /* Not stderr_out(), which does additional things and may trigger
3404          a recursive assertion failure.  fprintf was undeffed above, in
3405          case it was encapsulated. */
3406       fprintf (stderr,
3407                "Fatal error: recursive assertion failure, "
3408                "file %s, line %d, %s\n",
3409                file, line, expr);
3410       fprintf (stderr,
3411                "Original assertion failure: file %s, line %d, %s\n",
3412                assert_failed_file, assert_failed_line, assert_failed_expr);
3413     }
3414   else
3415     {
3416       assert_failed_file = file;
3417       assert_failed_line = line;
3418       assert_failed_expr = expr;
3419
3420       if (!initialized)
3421         fprintf (stderr,
3422                  "Fatal error: assertion failed, file %s, line %d, %s\n",
3423                  file, line, expr);
3424       else
3425         stderr_out ("Fatal error: assertion failed, file %s, line %d, %s\n",
3426                     file, line, expr);
3427     }
3428
3429   enter_debugger ();
3430 #if !defined (ASSERTIONS_DONT_ABORT)
3431   abort ();
3432 #endif
3433   inhibit_non_essential_printing_operations = 0;
3434   in_assert_failed = 0;
3435 }
3436 #endif /* USE_ASSERTIONS */
3437
3438 #ifdef QUANTIFY
3439 DEFUN ("quantify-start-recording-data", Fquantify_start_recording_data,
3440        0, 0, "", /*
3441 Start recording Quantify data.
3442 */
3443        ())
3444 {
3445   quantify_start_recording_data ();
3446   return Qnil;
3447 }
3448
3449 DEFUN ("quantify-stop-recording-data", Fquantify_stop_recording_data,
3450        0, 0, "", /*
3451 Stop recording Quantify data.
3452 */
3453        ())
3454 {
3455   quantify_stop_recording_data ();
3456   return Qnil;
3457 }
3458
3459 DEFUN ("quantify-clear-data", Fquantify_clear_data, 0, 0, "", /*
3460 Clear all Quantify data.
3461 */
3462        ())
3463 {
3464   quantify_clear_data ();
3465   return Qnil;
3466 }
3467 #endif /* QUANTIFY */
3468
3469 void
3470 syms_of_emacs (void)
3471 {
3472 #ifndef CANNOT_DUMP
3473   DEFSUBR (Fdump_emacs);
3474 #endif /* !CANNOT_DUMP */
3475
3476   DEFSUBR (Frun_emacs_from_temacs);
3477   DEFSUBR (Frunning_temacs_p);
3478   DEFSUBR (Finvocation_name);
3479   DEFSUBR (Finvocation_directory);
3480   DEFSUBR (Fkill_emacs);
3481   DEFSUBR (Fnoninteractive);
3482
3483 #ifdef DEBUG_XEMACS
3484   DEFSUBR (Fforce_debugging_signal);
3485 #endif
3486
3487 #ifdef QUANTIFY
3488   DEFSUBR (Fquantify_start_recording_data);
3489   DEFSUBR (Fquantify_stop_recording_data);
3490   DEFSUBR (Fquantify_clear_data);
3491 #endif /* QUANTIFY */
3492
3493   DEFSUBR (Fsplit_string_by_char);
3494   DEFSUBR (Fsplit_path);        /* #### */
3495
3496   defsymbol (&Qkill_emacs_hook, "kill-emacs-hook");
3497   defsymbol (&Qsave_buffers_kill_emacs, "save-buffers-kill-emacs");
3498 }
3499
3500 void
3501 vars_of_emacs (void)
3502 {
3503   DEFVAR_BOOL ("suppress-early-error-handler-backtrace",
3504                &suppress_early_error_handler_backtrace /*
3505 Non-nil means early error handler shouldn't print a backtrace.
3506 */ );
3507
3508   DEFVAR_LISP ("command-line-args", &Vcommand_line_args /*
3509 Args passed by shell to XEmacs, as a list of strings.
3510 */ );
3511
3512   DEFVAR_LISP ("invocation-name", &Vinvocation_name /*
3513 The program name that was used to run XEmacs.
3514 Any directory names are omitted.
3515 */ );
3516
3517   DEFVAR_LISP ("invocation-directory", &Vinvocation_directory /*
3518 The directory in which the XEmacs executable was found, to run it.
3519 The value is simply the program name if that directory's name is not known.
3520 */ );
3521
3522   DEFVAR_LISP ("invocation-path", &Vinvocation_path /*
3523 The path in which the XEmacs executable was found, to run it.
3524 The value is simply the value of environment variable PATH on startup
3525 if XEmacs was found there.
3526 */ );
3527
3528 #if 0 /* FSFmacs */
3529   xxDEFVAR_LISP ("installation-directory", &Vinstallation_directory,
3530     "A directory within which to look for the `lib-src' and `etc' directories.\n"
3531 "This is non-nil when we can't find those directories in their standard\n"
3532 "installed locations, but we can find them\n"
3533 "near where the XEmacs executable was found.");
3534 #endif
3535
3536   DEFVAR_LISP ("system-type", &Vsystem_type /*
3537 Symbol indicating type of operating system you are using.
3538 */ );
3539   Vsystem_type = intern (SYSTEM_TYPE);
3540   Fprovide (intern(SYSTEM_TYPE));
3541
3542 #ifndef EMACS_CONFIGURATION
3543 # define EMACS_CONFIGURATION "UNKNOWN"
3544 #endif
3545   DEFVAR_LISP ("system-configuration", &Vsystem_configuration /*
3546 String naming the configuration XEmacs was built for.
3547 */ );
3548   Vsystem_configuration = build_string (EMACS_CONFIGURATION);
3549
3550 #ifndef EMACS_CONFIG_OPTIONS
3551 # define EMACS_CONFIG_OPTIONS "UNKNOWN"
3552 #endif
3553   DEFVAR_LISP ("system-configuration-options", &Vsystem_configuration_options /*
3554 String containing the configuration options XEmacs was built with.
3555 */ );
3556   Vsystem_configuration_options = build_string (EMACS_CONFIG_OPTIONS);
3557
3558   DEFVAR_LISP ("emacs-major-version", &Vemacs_major_version /*
3559 Major version number of this version of Emacs, as an integer.
3560 Warning: this variable did not exist in Emacs versions earlier than:
3561   FSF Emacs:   19.23
3562   XEmacs:      19.10
3563 */ );
3564   Vemacs_major_version = make_int (EMACS_MAJOR_VERSION);
3565
3566   DEFVAR_LISP ("emacs-minor-version", &Vemacs_minor_version /*
3567 Minor version number of this version of Emacs, as an integer.
3568 Warning: this variable did not exist in Emacs versions earlier than:
3569   FSF Emacs:   19.23
3570   XEmacs:      19.10
3571 */ );
3572   Vemacs_minor_version = make_int (EMACS_MINOR_VERSION);
3573
3574   DEFVAR_LISP ("emacs-patch-level", &Vemacs_patch_level /*
3575 The patch level of this version of Emacs, as an integer.
3576 The value is non-nil if this version of XEmacs is part of a series of
3577 stable XEmacsen, but has bug fixes applied.
3578 Warning: this variable does not exist in FSF Emacs or in XEmacs versions
3579 earlier than 21.1.1
3580 */ );
3581 #ifdef EMACS_PATCH_LEVEL
3582   Vemacs_patch_level = make_int (EMACS_PATCH_LEVEL);
3583 #else
3584   Vemacs_patch_level = Qnil;
3585 #endif
3586
3587     DEFVAR_LISP ("emacs-beta-version", &Vemacs_beta_version /*
3588 Beta number of this version of Emacs, as an integer.
3589 The value is nil if this is an officially released version of XEmacs.
3590 Warning: this variable does not exist in FSF Emacs or in XEmacs versions
3591 earlier than 20.3.
3592 */ );
3593 #ifdef EMACS_BETA_VERSION
3594   Vemacs_beta_version = make_int (EMACS_BETA_VERSION);
3595 #else
3596   Vemacs_beta_version = Qnil;
3597 #endif
3598
3599 #ifdef INFODOCK
3600   DEFVAR_LISP ("infodock-major-version", &Vinfodock_major_version /*
3601 Major version number of this InfoDock release.
3602 */ );
3603   Vinfodock_major_version = make_int (INFODOCK_MAJOR_VERSION);
3604
3605   DEFVAR_LISP ("infodock-minor-version", &Vinfodock_minor_version /*
3606 Minor version number of this InfoDock release.
3607 */ );
3608   Vinfodock_minor_version = make_int (INFODOCK_MINOR_VERSION);
3609
3610   DEFVAR_LISP ("infodock-build-version", &Vinfodock_build_version /*
3611 Build version of this InfoDock release.
3612 */ );
3613   Vinfodock_build_version = make_int (INFODOCK_BUILD_VERSION);
3614 #endif
3615
3616   DEFVAR_LISP ("xemacs-codename", &Vxemacs_codename /*
3617 Codename of this version of Emacs (a string).
3618 */ );
3619 #ifndef XEMACS_CODENAME
3620 #define XEMACS_CODENAME "Noname"
3621 #endif
3622 #ifdef MULE
3623   {
3624     char dest[129];
3625     char src[64] = XEMACS_CODENAME;
3626     unsigned char* sp = (unsigned char*)src;
3627     int i = 0, chr;
3628
3629     while ( (chr = *sp++) && (i < 128) )
3630       {
3631         if (chr <= 0x7F)
3632           dest[i++] = chr;
3633         else
3634           {
3635 #ifdef UTF2000
3636             dest[i++] = (chr >> 6) | 0xC0;
3637             dest[i++] = (chr & 0x3F) | 0x80;
3638 #else
3639             dest[i++] = LEADING_BYTE_LATIN_ISO8859_1;
3640             dest[i++] = chr;
3641 #endif
3642           }
3643       }
3644     dest[i] = 0;
3645     Vxemacs_codename = build_string (dest);
3646   }
3647 #else  
3648   Vxemacs_codename = build_string (XEMACS_CODENAME);
3649 #endif
3650
3651   /* Lisp variables which contain command line flags.
3652
3653      The portable dumper stomps on these; they must be saved and restored
3654      if they are processed before the call to pdump_load() in main_1().
3655   */
3656   DEFVAR_BOOL ("noninteractive", &noninteractive1 /*
3657 Non-nil means XEmacs is running without interactive terminal.
3658 */ );
3659
3660   DEFVAR_BOOL ("inhibit-early-packages", &inhibit_early_packages /*
3661 Set to non-nil when the early packages should not be respected at startup.
3662 */ );
3663
3664   DEFVAR_BOOL ("inhibit-autoloads", &inhibit_autoloads /*
3665 Set to non-nil when autoloads should not be loaded at startup.
3666 */ );
3667
3668   DEFVAR_BOOL ("debug-paths", &debug_paths /*
3669 Set to non-nil when debug information about paths should be printed.
3670 */ );
3671
3672   DEFVAR_BOOL ("inhibit-site-lisp", &inhibit_site_lisp /*
3673 Set to non-nil when the site-lisp should not be searched at startup.
3674 */ );
3675 #ifdef INHIBIT_SITE_LISP
3676   inhibit_site_lisp = 1;
3677 #endif
3678
3679   DEFVAR_BOOL ("inhibit-site-modules", &inhibit_site_modules /*
3680 Set to non-nil when site-modules should not be searched at startup.
3681 */ );
3682 #ifdef INHIBIT_SITE_MODULES
3683   inhibit_site_modules = 1;
3684 #endif
3685
3686   DEFVAR_INT ("emacs-priority", &emacs_priority /*
3687 Priority for XEmacs to run at.
3688 This value is effective only if set before XEmacs is dumped,
3689 and only if the XEmacs executable is installed with setuid to permit
3690 it to change priority.  (XEmacs sets its uid back to the real uid.)
3691 Currently, you need to define SET_EMACS_PRIORITY in `config.h'
3692 before you compile XEmacs, to enable the code for this feature.
3693 */ );
3694   emacs_priority = 0;
3695
3696   DEFVAR_CONST_LISP ("internal-error-checking", &Vinternal_error_checking /*
3697 Internal error checking built-in into this instance of XEmacs.
3698 This is a list of symbols, initialized at build-time.  Legal symbols
3699 are:
3700
3701 extents         - check extents prior to each extent change;
3702 typecheck       - check types strictly, aborting in case of error;
3703 malloc          - check operation of malloc;
3704 gc              - check garbage collection;
3705 bufpos          - check buffer positions.
3706
3707 quick-build     - user has requested the "quick-build" configure option.
3708 */ );
3709   Vinternal_error_checking = Qnil;
3710 #ifdef ERROR_CHECK_EXTENTS
3711   Vinternal_error_checking = Fcons (intern ("extents"),
3712                                     Vinternal_error_checking);
3713 #endif
3714 #ifdef ERROR_CHECK_TYPECHECK
3715   Vinternal_error_checking = Fcons (intern ("typecheck"),
3716                                     Vinternal_error_checking);
3717 #endif
3718 #ifdef ERROR_CHECK_MALLOC
3719   Vinternal_error_checking = Fcons (intern ("malloc"),
3720                                     Vinternal_error_checking);
3721 #endif
3722 #ifdef ERROR_CHECK_GC
3723   Vinternal_error_checking = Fcons (intern ("gc"),
3724                                     Vinternal_error_checking);
3725 #endif
3726 #ifdef ERROR_CHECK_BUFPOS
3727   Vinternal_error_checking = Fcons (intern ("bufpos"),
3728                                     Vinternal_error_checking);
3729 #endif
3730 #ifdef QUICK_BUILD
3731   Vinternal_error_checking = Fcons (intern ("quick-build"),
3732                                     Vinternal_error_checking);
3733 #endif
3734
3735   DEFVAR_CONST_LISP ("mail-lock-methods", &Vmail_lock_methods /*
3736 Mail spool locking methods supported by this instance of XEmacs.
3737 This is a list of symbols.  Each of the symbols is one of the
3738 following: dot, lockf, flock, locking, mmdf.
3739 */ );
3740   {
3741     Vmail_lock_methods = Qnil;
3742     Vmail_lock_methods = Fcons (intern ("dot"), Vmail_lock_methods);
3743 #ifdef HAVE_LOCKF
3744     Vmail_lock_methods = Fcons (intern ("lockf"), Vmail_lock_methods);
3745 #endif
3746 #ifdef HAVE_FLOCK
3747     Vmail_lock_methods = Fcons (intern ("flock"), Vmail_lock_methods);
3748 #endif
3749 #ifdef HAVE_MMDF
3750     Vmail_lock_methods = Fcons (intern ("mmdf"), Vmail_lock_methods);
3751 #endif
3752 #ifdef HAVE_LOCKING
3753     Vmail_lock_methods = Fcons (intern ("locking"), Vmail_lock_methods);
3754 #endif
3755   }
3756
3757   DEFVAR_CONST_LISP ("configure-mail-lock-method", &Vconfigure_mail_lock_method /*
3758 Mail spool locking method suggested by configure.  This is one
3759 of the symbols in MAIL-LOCK-METHODS.
3760 */ );
3761   {
3762 #if defined(MAIL_LOCK_FLOCK) && defined(HAVE_FLOCK)
3763     Vconfigure_mail_lock_method = intern("flock");
3764 #elif defined(MAIL_LOCK_LOCKF) && defined(HAVE_LOCKF)
3765     Vconfigure_mail_lock_method = intern("lockf");
3766 #elif defined(MAIL_LOCK_MMDF) && defined(HAVE_MMDF)
3767     Vconfigure_mail_lock_method = intern("mmdf");
3768 #elif defined(MAIL_LOCK_LOCKING) && defined(HAVE_LOCKING)
3769     Vconfigure_mail_lock_method = intern("locking");
3770 #else
3771     Vconfigure_mail_lock_method = intern("dot");
3772 #endif
3773   }
3774
3775   DEFVAR_LISP ("path-separator", &Vpath_separator /*
3776 The directory separator in search paths, as a string.
3777 */ );
3778   {
3779     char c = SEPCHAR;
3780     Vpath_separator = make_string ((Bufbyte *)&c, 1);
3781   }
3782 }
3783
3784 void
3785 complex_vars_of_emacs (void)
3786 {
3787   /* This is all related to path searching. */
3788
3789   DEFVAR_LISP ("emacs-program-name", &Vemacs_program_name /*
3790 *Name of the Emacs variant.
3791 For example, this may be \"xemacs\" or \"infodock\".
3792 This is mainly meant for use in path searching.
3793 */ );
3794   Vemacs_program_name = build_string ((char *) PATH_PROGNAME);
3795
3796   DEFVAR_LISP ("emacs-program-version", &Vemacs_program_version /*
3797 *Version of the Emacs variant.
3798 This typically has the form NN.NN-bNN.
3799 This is mainly meant for use in path searching.
3800 */ );
3801   Vemacs_program_version = build_string ((char *) PATH_VERSION);
3802
3803   DEFVAR_LISP ("exec-path", &Vexec_path /*
3804 *List of directories to search programs to run in subprocesses.
3805 Each element is a string (directory name) or nil (try default directory).
3806 */ );
3807   Vexec_path = Qnil;
3808
3809   DEFVAR_LISP ("exec-directory", &Vexec_directory /*
3810 *Directory of architecture-dependent files that come with XEmacs,
3811 especially executable programs intended for XEmacs to invoke.
3812 */ );
3813   Vexec_directory = Qnil;
3814
3815   DEFVAR_LISP ("configure-exec-directory", &Vconfigure_exec_directory /*
3816 For internal use by the build procedure only.
3817 configure's idea of what `exec-directory' will be.
3818 */ );
3819 #ifdef PATH_EXEC
3820   Vconfigure_exec_directory = Ffile_name_as_directory
3821     (build_string ((char *) PATH_EXEC));
3822 #else
3823   Vconfigure_exec_directory = Qnil;
3824 #endif
3825
3826   DEFVAR_LISP ("lisp-directory", &Vlisp_directory /*
3827 *Directory of core Lisp files that come with XEmacs.
3828 */ );
3829   Vlisp_directory = Qnil;
3830
3831   DEFVAR_LISP ("configure-lisp-directory", &Vconfigure_lisp_directory /*
3832 For internal use by the build procedure only.
3833 configure's idea of what `lisp-directory' will be.
3834 */ );
3835 #ifdef PATH_LOADSEARCH
3836   Vconfigure_lisp_directory = Ffile_name_as_directory
3837     (build_string ((char *) PATH_LOADSEARCH));
3838 #else
3839   Vconfigure_lisp_directory = Qnil;
3840 #endif
3841
3842   DEFVAR_LISP ("mule-lisp-directory", &Vmule_lisp_directory /*
3843 *Directory of Mule Lisp files that come with XEmacs.
3844 */ );
3845   Vmule_lisp_directory = Qnil;
3846
3847   DEFVAR_LISP ("configure-mule-lisp-directory", &Vconfigure_mule_lisp_directory /*
3848 For internal use by the build procedure only.
3849 configure's idea of what `mule-lisp-directory' will be.
3850 */ );
3851 #ifdef PATH_MULELOADSEARCH
3852   Vconfigure_mule_lisp_directory = Ffile_name_as_directory
3853     (build_string ((char *) PATH_MULELOADSEARCH));
3854 #else
3855   Vconfigure_mule_lisp_directory = Qnil;
3856 #endif
3857
3858   DEFVAR_LISP ("utf-2000-lisp-directory", &Vutf_2000_lisp_directory /*
3859 *Directory of UTF-2000 Lisp files that come with XEmacs.
3860 */ );
3861   Vutf_2000_lisp_directory = Qnil;
3862
3863   DEFVAR_LISP ("configure-utf-2000-lisp-directory",
3864                &Vconfigure_utf_2000_lisp_directory /*
3865 For internal use by the build procedure only.
3866 configure's idea of what `utf-2000-lisp-directory' will be.
3867 */ );
3868 #ifdef PATH_UTF2000LOADSEARCH
3869   Vconfigure_utf_2000_lisp_directory = Ffile_name_as_directory
3870     (build_string ((char *) PATH_UTF2000LOADSEARCH));
3871 #else
3872   Vconfigure_utf_2000_lisp_directory = Qnil;
3873 #endif
3874
3875   DEFVAR_LISP ("module-directory", &Vmodule_directory /*
3876 *Directory of core dynamic modules that come with XEmacs.
3877 */ );
3878   Vmodule_directory = Qnil;
3879
3880   DEFVAR_LISP ("configure-module-directory", &Vconfigure_module_directory /*
3881 For internal use by the build procedure only.
3882 configure's idea of what `module-directory' will be.
3883 */ );
3884 #ifdef PATH_MODULESEARCH
3885   Vconfigure_module_directory = Ffile_name_as_directory
3886     (build_string ((char *) PATH_MODULESEARCH));
3887 #else
3888   Vconfigure_module_directory = Qnil;
3889 #endif
3890
3891   DEFVAR_LISP ("configure-package-path", &Vconfigure_package_path /*
3892 For internal use by the build procedure only.
3893 configure's idea of what the package path will be.
3894 */ );
3895 #ifdef PATH_PACKAGEPATH
3896   Vconfigure_package_path = decode_path (PATH_PACKAGEPATH);
3897 #else
3898   Vconfigure_package_path = Qnil;
3899 #endif
3900
3901   DEFVAR_LISP ("data-directory", &Vdata_directory /*
3902 *Directory of architecture-independent files that come with XEmacs,
3903 intended for XEmacs to use.
3904 Use of this variable in new code is almost never correct.  See the
3905 functions `locate-data-file' and `locate-data-directory' and the variable
3906 `data-directory-list'.
3907 */ );
3908   Vdata_directory = Qnil;
3909
3910   DEFVAR_LISP ("configure-data-directory", &Vconfigure_data_directory /*
3911 For internal use by the build procedure only.
3912 configure's idea of what `data-directory' will be.
3913 */ );
3914 #ifdef PATH_DATA
3915   Vconfigure_data_directory = Ffile_name_as_directory
3916     (build_string ((char *) PATH_DATA));
3917 #else
3918   Vconfigure_data_directory = Qnil;
3919 #endif
3920
3921   DEFVAR_LISP ("data-directory-list", &Vdata_directory_list /*
3922 *List of directories of architecture-independent files that come with XEmacs
3923 or were installed as packages, and are intended for XEmacs to use.
3924 */ );
3925   Vdata_directory_list = Qnil;
3926
3927   DEFVAR_LISP ("site-directory", &Vsite_directory /*
3928 *Directory of site-specific Lisp files that come with XEmacs.
3929 */ );
3930   Vsite_directory = Qnil;
3931
3932   DEFVAR_LISP ("configure-site-directory", &Vconfigure_site_directory /*
3933 For internal use by the build procedure only.
3934 configure's idea of what `site-directory' will be.
3935 */ );
3936 #ifdef PATH_SITE
3937   Vconfigure_site_directory = Ffile_name_as_directory
3938     (build_string ((char *) PATH_SITE));
3939 #else
3940   Vconfigure_site_directory = Qnil;
3941 #endif
3942
3943   DEFVAR_LISP ("site-module-directory", &Vsite_module_directory /*
3944 *Directory of site-specific loadable modules that come with XEmacs.
3945 */ );
3946   Vsite_module_directory = Qnil;
3947
3948   DEFVAR_LISP ("configure-site-module-directory", &Vconfigure_site_module_directory /*
3949 For internal use by the build procedure only.
3950 configure's idea of what `site-directory' will be.
3951 */ );
3952 #ifdef PATH_SITE_MODULES
3953   Vconfigure_site_module_directory = Ffile_name_as_directory
3954     (build_string ((char *) PATH_SITE_MODULES));
3955 #else
3956   Vconfigure_site_module_directory = Qnil;
3957 #endif
3958
3959   DEFVAR_LISP ("doc-directory", &Vdoc_directory /*
3960 *Directory containing the DOC file that comes with XEmacs.
3961 This is usually the same as `exec-directory'.
3962 */ );
3963   Vdoc_directory = Qnil;
3964
3965   DEFVAR_LISP ("configure-doc-directory", &Vconfigure_doc_directory /*
3966 For internal use by the build procedure only.
3967 configure's idea of what `doc-directory' will be.
3968 */ );
3969 #ifdef PATH_DOC
3970   Vconfigure_doc_directory = Ffile_name_as_directory
3971     (build_string ((char *) PATH_DOC));
3972 #else
3973   Vconfigure_doc_directory = Qnil;
3974 #endif
3975
3976   DEFVAR_LISP ("configure-exec-prefix-directory", &Vconfigure_exec_prefix_directory /*
3977 For internal use by the build procedure only.
3978 configure's idea of what `exec-prefix-directory' will be.
3979 */ );
3980 #ifdef PATH_EXEC_PREFIX
3981   Vconfigure_exec_prefix_directory = Ffile_name_as_directory
3982     (build_string ((char *) PATH_EXEC_PREFIX));
3983 #else
3984   Vconfigure_exec_prefix_directory = Qnil;
3985 #endif
3986
3987   DEFVAR_LISP ("configure-prefix-directory", &Vconfigure_prefix_directory /*
3988 For internal use by the build procedure only.
3989 configure's idea of what `prefix-directory' will be.
3990 */ );
3991 #ifdef PATH_PREFIX
3992   Vconfigure_prefix_directory = Ffile_name_as_directory
3993     (build_string ((char *) PATH_PREFIX));
3994 #else
3995   Vconfigure_prefix_directory = Qnil;
3996 #endif
3997
3998   DEFVAR_LISP ("configure-info-directory", &Vconfigure_info_directory /*
3999 For internal use by the build procedure only.
4000 This is the name of the directory in which the build procedure installed
4001 Emacs's info files; the default value for Info-default-directory-list
4002 includes this.
4003 */ );
4004 #ifdef PATH_INFO
4005   Vconfigure_info_directory =
4006     Ffile_name_as_directory (build_string (PATH_INFO));
4007 #else
4008   Vconfigure_info_directory = Qnil;
4009 #endif
4010
4011   DEFVAR_LISP ("configure-info-path", &Vconfigure_info_path /*
4012 The configured initial path for info documentation.
4013 */ );
4014 #ifdef PATH_INFOPATH
4015   Vconfigure_info_path = decode_path (PATH_INFOPATH);
4016 #else
4017   Vconfigure_info_path = Qnil;
4018 #endif
4019 }
4020
4021 #if defined(__sgi) && !defined(PDUMP)
4022 /* This is so tremendously ugly I'd puke. But then, it works.
4023  * The target is to override the static constructor from the
4024  * libiflPNG.so library which is masquerading as libz, and
4025  * cores on us when re-started from the dumped executable.
4026  * This will have to go for 21.1  -- OG.
4027  */
4028 void __sti__iflPNGFile_c___ (void);
4029 void
4030 __sti__iflPNGFile_c___ (void)
4031 {
4032 }
4033
4034 #endif