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