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