[__GNUC__] (C_DEBUG_SWITCH): New definition.
[bpt/emacs.git] / src / emacs.c
CommitLineData
f927c5ae 1/* Fully extensible Emacs, running on Unix, intended for GNU.
f8c25f1b 2 Copyright (C) 1985, 86, 87, 93, 94, 95 Free Software Foundation, Inc.
f927c5ae
JB
3
4This file is part of GNU Emacs.
5
6GNU Emacs is free software; you can redistribute it and/or modify
7it under the terms of the GNU General Public License as published by
40be253a 8the Free Software Foundation; either version 2, or (at your option)
f927c5ae
JB
9any later version.
10
11GNU Emacs is distributed in the hope that it will be useful,
12but WITHOUT ANY WARRANTY; without even the implied warranty of
13MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14GNU General Public License for more details.
15
16You should have received a copy of the GNU General Public License
17along with GNU Emacs; see the file COPYING. If not, write to
5a7670bf
RS
18the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
19Boston, MA 02111-1307, USA. */
f927c5ae
JB
20
21
22#include <signal.h>
23#include <errno.h>
24
18160b98 25#include <config.h>
f927c5ae
JB
26#include <stdio.h>
27
28#include <sys/types.h>
29#include <sys/file.h>
30
31#ifdef VMS
32#include <ssdef.h>
33#endif
34
f927c5ae
JB
35#ifdef BSD
36#include <sys/ioctl.h>
37#endif
38
f927c5ae
JB
39#include "lisp.h"
40#include "commands.h"
bef79ee4 41#include "intervals.h"
f927c5ae 42
edc8ae07 43#include "systty.h"
6c3a4e9d 44#include "blockinput.h"
8090eb09 45#include "syssignal.h"
6c362a8b 46#include "process.h"
a41f8bed 47
f927c5ae
JB
48#ifndef O_RDWR
49#define O_RDWR 2
50#endif
51
43165242 52extern void malloc_warning ();
0269dedb 53extern void set_time_zone_rule ();
43165242
KH
54extern char *index ();
55extern char *strerror ();
56
f927c5ae
JB
57/* Command line args from shell, as list of strings */
58Lisp_Object Vcommand_line_args;
59
59653951
JB
60/* The name under which Emacs was invoked, with any leading directory
61 names discarded. */
62Lisp_Object Vinvocation_name;
63
ace40a69
RS
64/* The directory name from which Emacs was invoked. */
65Lisp_Object Vinvocation_directory;
66
07f4d123
RS
67/* The directory name in which to find subdirs such as lisp and etc.
68 nil means get them only from PATH_LOADSEARCH. */
69Lisp_Object Vinstallation_directory;
70
e5d77022
JB
71/* Hook run by `kill-emacs' before it does really anything. */
72Lisp_Object Vkill_emacs_hook;
73
f927c5ae
JB
74/* Set nonzero after Emacs has started up the first time.
75 Prevents reinitialization of the Lisp world and keymaps
76 on subsequent starts. */
77int initialized;
78
271c7b7c 79/* Variable whose value is symbol giving operating system type. */
f927c5ae 80Lisp_Object Vsystem_type;
271c7b7c
RS
81
82/* Variable whose value is string giving configuration built for. */
83Lisp_Object Vsystem_configuration;
f0fc0b1a
KH
84
85/* Variable whose value is string giving configuration options,
86 for use when reporting bugs. */
87Lisp_Object Vsystem_configuration_options;
88
f927c5ae
JB
89/* If non-zero, emacs should not attempt to use an window-specific code,
90 but instead should use the virtual terminal under which it was started */
91int inhibit_window_system;
92
5aa7f46a
JB
93/* If nonzero, set Emacs to run at this priority. This is also used
94 in child_setup and sys_suspend to make sure subshells run at normal
95 priority; Those functions have their own extern declaration. */
3005da00
RS
96int emacs_priority;
97
7074fde6
FP
98/* If non-zero a filter or a sentinel is running. Tested to save the match
99 data on the first attempt to change it inside asynchronous code. */
100int running_asynch_code;
101
647cec74 102#ifdef BSD_PGRPS
9ae8f997
JB
103/* See sysdep.c. */
104extern int inherited_pgroup;
105#endif
106
f927c5ae
JB
107#ifdef HAVE_X_WINDOWS
108/* If non-zero, -d was specified, meaning we're using some window system. */
109int display_arg;
110#endif
111
112/* An address near the bottom of the stack.
113 Tells GC how to save a copy of the stack. */
114char *stack_bottom;
115
8ba50e1a 116#ifdef HAVE_WINDOW_SYSTEM
f927c5ae 117extern Lisp_Object Vwindow_system;
8ba50e1a 118#endif /* HAVE_WINDOW_SYSTEM */
f927c5ae 119
1090a161
RS
120extern Lisp_Object Vauto_save_list_file_name;
121
f927c5ae
JB
122#ifdef USG_SHARED_LIBRARIES
123/* If nonzero, this is the place to put the end of the writable segment
124 at startup. */
125
126unsigned int bss_end = 0;
127#endif
128
129/* Nonzero means running Emacs without interactive terminal. */
130
131int noninteractive;
132
133/* Value of Lisp variable `noninteractive'.
134 Normally same as C variable `noninteractive'
135 but nothing terrible happens if user sets this one. */
136
137int noninteractive1;
e29f86e4
RS
138
139/* Save argv and argc. */
140char **initial_argv;
141int initial_argc;
081bef73
RS
142
143static void sort_args ();
f927c5ae
JB
144\f
145/* Signal code for the fatal signal that was received */
146int fatal_error_code;
147
148/* Nonzero if handling a fatal error already */
149int fatal_error_in_progress;
150
151/* Handle bus errors, illegal instruction, etc. */
2447c626 152SIGTYPE
f927c5ae
JB
153fatal_error_signal (sig)
154 int sig;
155{
f927c5ae
JB
156 fatal_error_code = sig;
157 signal (sig, SIG_DFL);
158
061b7f94
RS
159 TOTALLY_UNBLOCK_INPUT;
160
f927c5ae 161 /* If fatal error occurs in code below, avoid infinite recursion. */
8090eb09
JB
162 if (! fatal_error_in_progress)
163 {
164 fatal_error_in_progress = 1;
f927c5ae 165
f7ab4e3d 166 shut_down_emacs (sig, 0, Qnil);
8090eb09 167 }
f927c5ae
JB
168
169#ifdef VMS
f927c5ae
JB
170 LIB$STOP (SS$_ABORT);
171#else
8090eb09
JB
172 /* Signal the same code; this time it will really be fatal.
173 Remember that since we're in a signal handler, the signal we're
174 going to send is probably blocked, so we have to unblock it if we
175 want to really receive it. */
29b89fe0 176#ifndef MSDOS
a90538cb 177 sigunblock (sigmask (fatal_error_code));
29b89fe0 178#endif
f927c5ae
JB
179 kill (getpid (), fatal_error_code);
180#endif /* not VMS */
181}
271c7b7c
RS
182
183#ifdef SIGDANGER
184
16c323ee 185/* Handler for SIGDANGER. */
271c7b7c
RS
186SIGTYPE
187memory_warning_signal (sig)
188 int sig;
189{
190 signal (sig, memory_warning_signal);
191
192 malloc_warning ("Operating system warns that virtual memory is running low.\n");
55796183
RS
193
194 /* It might be unsafe to call do_auto_save now. */
195 force_auto_save_soon ();
271c7b7c
RS
196}
197#endif
f927c5ae
JB
198\f
199/* Code for dealing with Lisp access to the Unix command line */
200
201static
202init_cmdargs (argc, argv, skip_args)
203 int argc;
204 char **argv;
205 int skip_args;
206{
207 register int i;
07f4d123 208 Lisp_Object name, dir;
f927c5ae 209
e29f86e4
RS
210 initial_argv = argv;
211 initial_argc = argc;
212
cb421be8 213 Vinvocation_name = Ffile_name_nondirectory (build_string (argv[0]));
ace40a69
RS
214 Vinvocation_directory = Ffile_name_directory (build_string (argv[0]));
215 /* If we got no directory in argv[0], search PATH to find where
216 Emacs actually came from. */
217 if (NILP (Vinvocation_directory))
218 {
219 Lisp_Object found;
220 int yes = openp (Vexec_path, Vinvocation_name,
6c362a8b 221 EXEC_SUFFIXES, &found, 1);
e443f843 222 if (yes == 1)
ace40a69
RS
223 Vinvocation_directory = Ffile_name_directory (found);
224 }
59653951 225
4133b300
RS
226 if (!NILP (Vinvocation_directory)
227 && NILP (Ffile_name_absolute_p (Vinvocation_directory)))
228 /* Emacs was started with relative path, like ./emacs */
229 Vinvocation_directory = Fexpand_file_name (Vinvocation_directory, Qnil);
230
07f4d123
RS
231 Vinstallation_directory = Qnil;
232
233 if (!NILP (Vinvocation_directory))
234 {
235 dir = Vinvocation_directory;
236 name = Fexpand_file_name (Vinvocation_name, dir);
237 while (1)
238 {
f5ab9736 239 Lisp_Object tem, lib_src_exists;
07f4d123
RS
240 Lisp_Object etc_exists, info_exists;
241
f5ab9736
RS
242 /* See if dir contains subdirs for use by Emacs.
243 Check for the ones that would exist in a build directory,
244 not including lisp and info. */
245 tem = Fexpand_file_name (build_string ("lib-src"), dir);
246 lib_src_exists = Ffile_exists_p (tem);
247 if (!NILP (lib_src_exists))
07f4d123 248 {
f5ab9736
RS
249 tem = Fexpand_file_name (build_string ("etc"), dir);
250 etc_exists = Ffile_exists_p (tem);
251 if (!NILP (etc_exists))
07f4d123 252 {
f5ab9736
RS
253 Vinstallation_directory
254 = Ffile_name_as_directory (dir);
255 break;
07f4d123
RS
256 }
257 }
258
259 /* See if dir's parent contains those subdirs. */
f5ab9736
RS
260 tem = Fexpand_file_name (build_string ("../lib-src"), dir);
261 lib_src_exists = Ffile_exists_p (tem);
262 if (!NILP (lib_src_exists))
07f4d123 263 {
f5ab9736
RS
264 tem = Fexpand_file_name (build_string ("../etc"), dir);
265 etc_exists = Ffile_exists_p (tem);
266 if (!NILP (etc_exists))
07f4d123 267 {
f5ab9736
RS
268 tem = Fexpand_file_name (build_string (".."), dir);
269 Vinstallation_directory
270 = Ffile_name_as_directory (tem);
271 break;
07f4d123
RS
272 }
273 }
274
275 /* If the Emacs executable is actually a link,
276 next try the dir that the link points into. */
277 tem = Ffile_symlink_p (name);
278 if (!NILP (tem))
279 {
260ec24d 280 name = Fexpand_file_name (tem, dir);
07f4d123
RS
281 dir = Ffile_name_directory (name);
282 }
283 else
284 break;
285 }
286 }
287
f927c5ae
JB
288 Vcommand_line_args = Qnil;
289
290 for (i = argc - 1; i >= 0; i--)
291 {
292 if (i == 0 || i > skip_args)
293 Vcommand_line_args
294 = Fcons (build_string (argv[i]), Vcommand_line_args);
295 }
296}
59653951
JB
297
298DEFUN ("invocation-name", Finvocation_name, Sinvocation_name, 0, 0, 0,
299 "Return the program name that was used to run Emacs.\n\
300Any directory names are omitted.")
301 ()
302{
303 return Fcopy_sequence (Vinvocation_name);
304}
305
ace40a69
RS
306DEFUN ("invocation-directory", Finvocation_directory, Sinvocation_directory,
307 0, 0, 0,
308 "Return the directory name in which the Emacs executable was located")
309 ()
310{
311 return Fcopy_sequence (Vinvocation_directory);
312}
313
f927c5ae
JB
314\f
315#ifdef VMS
316#ifdef LINK_CRTL_SHARE
34035df3 317#ifdef SHARABLE_LIB_BUG
f927c5ae 318extern noshare char **environ;
34035df3 319#endif /* SHARABLE_LIB_BUG */
f927c5ae
JB
320#endif /* LINK_CRTL_SHARE */
321#endif /* VMS */
322
0269dedb
RS
323#ifdef HAVE_TZSET
324/* A valid but unlikely value for the TZ environment value.
325 It is OK (though a bit slower) if the user actually chooses this value. */
326static char dump_tz[] = "UtC0";
327#endif
328
a90538cb 329#ifndef ORDINARY_LINK
efd241cc
RS
330/* We don't include crtbegin.o and crtend.o in the link,
331 so these functions and variables might be missed.
332 Provide dummy definitions to avoid error.
333 (We don't have any real constructors or destructors.) */
334#ifdef __GNUC__
46e65b73 335#ifndef GCC_CTORS_IN_LIBC
c83a7064 336__do_global_ctors ()
efd241cc 337{}
c83a7064
RS
338__do_global_ctors_aux ()
339{}
33143604
RS
340__do_global_dtors ()
341{}
64c1864a
RS
342/* Linux has a bug in its library; avoid an error. */
343#ifndef LINUX
c83a7064 344char * __CTOR_LIST__[2] = { (char *) (-1), 0 };
64c1864a 345#endif
c83a7064 346char * __DTOR_LIST__[2] = { (char *) (-1), 0 };
46e65b73 347#endif /* GCC_CTORS_IN_LIBC */
c83a7064 348__main ()
efd241cc 349{}
efd241cc 350#endif /* __GNUC__ */
a90538cb 351#endif /* ORDINARY_LINK */
efd241cc 352
e2925360
KH
353/* Test whether the next argument in ARGV matches SSTR or a prefix of
354 LSTR (at least MINLEN characters). If so, then if VALPTR is non-null
355 (the argument is supposed to have a value) store in *VALPTR either
356 the next argument or the portion of this one after the equal sign.
357 ARGV is read starting at position *SKIPPTR; this index is advanced
358 by the number of arguments used.
359
360 Too bad we can't just use getopt for all of this, but we don't have
361 enough information to do it right. */
081bef73 362
e2925360 363static int
df6530f8 364argmatch (argv, argc, sstr, lstr, minlen, valptr, skipptr)
e2925360 365 char **argv;
df6530f8 366 int argc;
e2925360
KH
367 char *sstr;
368 char *lstr;
369 int minlen;
370 char **valptr;
371 int *skipptr;
372{
373 char *p;
374 int arglen;
df6530f8
RS
375 char *arg;
376
377 /* Don't access argv[argc]; give up in advance. */
378 if (argc <= *skipptr + 1)
379 return 0;
380
381 arg = argv[*skipptr+1];
e2925360
KH
382 if (arg == NULL)
383 return 0;
384 if (strcmp (arg, sstr) == 0)
385 {
386 if (valptr != NULL)
387 {
388 *valptr = argv[*skipptr+2];
389 *skipptr += 2;
390 }
391 else
392 *skipptr += 1;
393 return 1;
394 }
395 arglen = (valptr != NULL && (p = index (arg, '=')) != NULL
396 ? p - arg : strlen (arg));
c03e1113 397 if (lstr == 0 || arglen < minlen || strncmp (arg, lstr, arglen) != 0)
e2925360
KH
398 return 0;
399 else if (valptr == NULL)
400 {
401 *skipptr += 1;
402 return 1;
403 }
404 else if (p != NULL)
405 {
406 *valptr = p+1;
407 *skipptr += 1;
408 return 1;
409 }
410 else if (argv[*skipptr+2] != NULL)
411 {
412 *valptr = argv[*skipptr+2];
413 *skipptr += 2;
414 return 1;
415 }
416 else
417 {
418 return 0;
419 }
420}
421
f927c5ae
JB
422/* ARGSUSED */
423main (argc, argv, envp)
424 int argc;
425 char **argv;
426 char **envp;
427{
428 char stack_bottom_variable;
429 int skip_args = 0;
430 extern int errno;
431 extern sys_nerr;
f927c5ae 432
6000fe37
RS
433#ifdef LINUX_SBRK_BUG
434 __sbrk (1);
435#endif
436
081bef73
RS
437 sort_args (argc, argv);
438
1702afef
RS
439 if (argmatch (argv, argc, "-version", "--version", 3, NULL, &skip_args))
440 {
441 Lisp_Object tem;
442 tem = Fsymbol_value (intern ("emacs-version"));
443 if (!STRINGP (tem))
444 {
445 fprintf (stderr, "Invalid value of `emacs-version'\n");
446 exit (1);
447 }
448 else
449 {
30ce1583 450 printf ("GNU Emacs %s\n", XSTRING (tem)->data);
1702afef
RS
451 exit (0);
452 }
453 }
454
f927c5ae
JB
455/* Map in shared memory, if we are using that. */
456#ifdef HAVE_SHM
df6530f8 457 if (argmatch (argv, argc, "-nl", "--no-shared-memory", 6, NULL, &skip_args))
f927c5ae
JB
458 {
459 map_in_data (0);
460 /* The shared memory was just restored, which clobbered this. */
461 skip_args = 1;
462 }
463 else
464 {
465 map_in_data (1);
466 /* The shared memory was just restored, which clobbered this. */
467 skip_args = 0;
468 }
469#endif
470
19a36ec6 471#ifdef NeXT
b9df9faa
KH
472 {
473 extern int malloc_cookie;
474 /* This helps out unexnext.c. */
475 if (initialized)
476 if (malloc_jumpstart (malloc_cookie) != 0)
477 printf ("malloc jumpstart failed!\n");
478 }
19a36ec6
RS
479#endif /* NeXT */
480
f927c5ae
JB
481#ifdef VMS
482 /* If -map specified, map the data file in */
e2925360
KH
483 {
484 char *file;
df6530f8 485 if (argmatch (argv, argc, "-map", "--map-data", 3, &mapin_file, &skip_args))
e2925360
KH
486 mapin_data (file);
487 }
f927c5ae
JB
488
489#ifdef LINK_CRTL_SHARE
34035df3 490#ifdef SHARABLE_LIB_BUG
f927c5ae
JB
491 /* Bletcherous shared libraries! */
492 if (!stdin)
493 stdin = fdopen (0, "r");
494 if (!stdout)
495 stdout = fdopen (1, "w");
496 if (!stderr)
497 stderr = fdopen (2, "w");
498 if (!environ)
499 environ = envp;
34035df3 500#endif /* SHARABLE_LIB_BUG */
f927c5ae
JB
501#endif /* LINK_CRTL_SHARE */
502#endif /* VMS */
503
504 /* Record (approximately) where the stack begins. */
505 stack_bottom = &stack_bottom_variable;
506
507#ifdef RUN_TIME_REMAP
508 if (initialized)
509 run_time_remap (argv[0]);
510#endif
511
512#ifdef USG_SHARED_LIBRARIES
513 if (bss_end)
1d233b80 514 brk ((void *)bss_end);
f927c5ae
JB
515#endif
516
517 clearerr (stdin);
9ae8f997 518
f927c5ae
JB
519#ifndef SYSTEM_MALLOC
520 if (! initialized)
9ac0d9e0
JB
521 {
522 /* Arrange to get warning messages as memory fills up. */
523 memory_warnings (0, malloc_warning);
524
525 /* Arrange to disable interrupt input while malloc and friends are
526 running. */
527 uninterrupt_malloc ();
528 }
f927c5ae
JB
529#endif /* not SYSTEM_MALLOC */
530
29b89fe0
RS
531#ifdef MSDOS
532 /* We do all file input/output as binary files. When we need to translate
533 newlines, we do that manually. */
534 _fmode = O_BINARY;
18198bb2
RS
535
536#if __DJGPP__ >= 2
537 if (!isatty (fileno (stdin)))
538 setmode (fileno (stdin), O_BINARY);
539 if (!isatty (fileno (stdout)))
540 {
541 fflush (stdout);
542 setmode (fileno (stdout), O_BINARY);
543 }
544#else /* not __DJGPP__ >= 2 */
29b89fe0
RS
545 (stdin)->_flag &= ~_IOTEXT;
546 (stdout)->_flag &= ~_IOTEXT;
547 (stderr)->_flag &= ~_IOTEXT;
18198bb2 548#endif /* not __DJGPP__ >= 2 */
29b89fe0
RS
549#endif /* MSDOS */
550
a422068f 551#ifdef SET_EMACS_PRIORITY
3005da00 552 if (emacs_priority)
5aa7f46a 553 nice (emacs_priority);
f927c5ae 554 setuid (getuid ());
a422068f 555#endif /* SET_EMACS_PRIORITY */
f927c5ae 556
d8b3a65d 557#ifdef EXTRA_INITIALIZE
a9260219 558 EXTRA_INITIALIZE;
d8b3a65d
RS
559#endif
560
f927c5ae
JB
561 inhibit_window_system = 0;
562
4fc0b45b 563 /* Handle the -t switch, which specifies filename to use as terminal */
e2925360
KH
564 {
565 char *term;
df6530f8 566 if (argmatch (argv, argc, "-t", "--terminal", 4, &term, &skip_args))
e2925360
KH
567 {
568 int result;
569 close (0);
570 close (1);
571 result = open (term, O_RDWR, 2 );
572 if (result < 0)
573 {
574 char *errstring = strerror (errno);
575 fprintf (stderr, "emacs: %s: %s\n", term, errstring);
576 exit (1);
577 }
578 dup (0);
579 if (! isatty (0))
580 {
581 fprintf (stderr, "emacs: %s: not a tty\n", term);
582 exit (1);
583 }
584 fprintf (stderr, "Using %s\n", term);
8ba50e1a 585#ifdef HAVE_WINDOW_SYSTEM
e2925360 586 inhibit_window_system = 1; /* -t => -nw */
f927c5ae 587#endif
e2925360
KH
588 }
589 }
df6530f8 590 if (argmatch (argv, argc, "-nw", "--no-windows", 6, NULL, &skip_args))
e2925360 591 inhibit_window_system = 1;
f927c5ae 592
e2925360 593 /* Handle the -batch switch, which means don't do interactive display. */
f927c5ae 594 noninteractive = 0;
df6530f8 595 if (argmatch (argv, argc, "-batch", "--batch", 5, NULL, &skip_args))
e2925360
KH
596 noninteractive = 1;
597
598 /* Handle the --help option, which gives a usage message.. */
df6530f8 599 if (argmatch (argv, argc, "-help", "--help", 3, NULL, &skip_args))
f927c5ae 600 {
e2925360
KH
601 printf ("\
602Usage: %s [-t term] [--terminal term] [-nw] [--no-windows] [--batch]\n\
603 [-q] [--no-init-file] [-u user] [--user user] [--debug-init]\n\
afaa660f 604 [--version] [--no-site-file]\n\
e2925360 605 [-f func] [--funcall func] [-l file] [--load file] [--insert file]\n\
afaa660f 606 [+linenum] file-to-visit [--kill]\n", argv[0]);
e2925360 607 exit (0);
f927c5ae
JB
608 }
609
081bef73
RS
610#ifdef HAVE_X_WINDOWS
611 /* Stupid kludge to catch command-line display spec. We can't
612 handle this argument entirely in window system dependent code
613 because we don't even know which window system dependent code
614 to run until we've recognized this argument. */
615 {
98ea0308 616 char *displayname = 0;
081bef73
RS
617 int i;
618 int count_before = skip_args;
619
df6530f8 620 if (argmatch (argv, argc, "-d", "--display", 3, &displayname, &skip_args))
081bef73 621 display_arg = 1;
df6530f8 622 else if (argmatch (argv, argc, "-display", 0, 3, &displayname, &skip_args))
081bef73
RS
623 display_arg = 1;
624
625 /* If we have the form --display=NAME,
626 convert it into -d name.
627 This requires inserting a new element into argv. */
628 if (displayname != 0 && skip_args - count_before == 1)
629 {
630 char **new = (char **) xmalloc (sizeof (char *) * (argc + 2));
631 int j;
632
633 for (j = 0; j < count_before + 1; j++)
634 new[j] = argv[j];
635 new[count_before + 1] = "-d";
636 new[count_before + 2] = displayname;
637 for (j = count_before + 2; j <argc; j++)
638 new[j + 1] = argv[j];
639 argv = new;
640 argc++;
641 }
1702afef
RS
642 /* Change --display to -d, when its arg is separate. */
643 else if (displayname != 0 && skip_args > count_before
644 && argv[count_before + 1][1] == '-')
813f79d5 645 argv[count_before + 1] = "-d";
081bef73
RS
646
647 /* Don't actually discard this arg. */
648 skip_args = count_before;
649 }
650#endif
651
edb85f59
RS
652 if (! noninteractive)
653 {
654#ifdef BSD_PGRPS
655 if (initialized)
656 {
657 inherited_pgroup = EMACS_GETPGRP (0);
658 setpgrp (0, getpid ());
659 }
660#else
661#if defined (USG5) && defined (INTERRUPT_INPUT)
662 setpgrp ();
663#endif
664#endif
665 }
666
fb8e9847
JB
667#ifdef POSIX_SIGNALS
668 init_signals ();
669#endif
670
1efa2983
KH
671 /* Don't catch SIGHUP if dumping. */
672 if (1
673#ifndef CANNOT_DUMP
674 && initialized
675#endif
676 )
677 {
57e3d22a 678 sigblock (sigmask (SIGHUP));
1efa2983
KH
679 /* In --batch mode, don't catch SIGHUP if already ignored.
680 That makes nohup work. */
681 if (! noninteractive
682 || signal (SIGHUP, SIG_IGN) != SIG_IGN)
683 signal (SIGHUP, fatal_error_signal);
57e3d22a 684 sigunblock (sigmask (SIGHUP));
1efa2983
KH
685 }
686
f927c5ae
JB
687 if (
688#ifndef CANNOT_DUMP
689 ! noninteractive || initialized
690#else
691 1
692#endif
693 )
694 {
1efa2983 695 /* Don't catch these signals in batch mode if dumping.
f927c5ae
JB
696 On some machines, this sets static data that would make
697 signal fail to work right when the dumped Emacs is run. */
f927c5ae
JB
698 signal (SIGQUIT, fatal_error_signal);
699 signal (SIGILL, fatal_error_signal);
700 signal (SIGTRAP, fatal_error_signal);
99e372cd
RS
701#ifdef SIGABRT
702 signal (SIGABRT, fatal_error_signal);
703#endif
704#ifdef SIGHWE
705 signal (SIGHWE, fatal_error_signal);
706#endif
707#ifdef SIGPRE
708 signal (SIGPRE, fatal_error_signal);
709#endif
710#ifdef SIGORE
711 signal (SIGORE, fatal_error_signal);
712#endif
713#ifdef SIGUME
714 signal (SIGUME, fatal_error_signal);
715#endif
716#ifdef SIGDLK
717 signal (SIGDLK, fatal_error_signal);
718#endif
719#ifdef SIGCPULIM
720 signal (SIGCPULIM, fatal_error_signal);
721#endif
a90538cb
JB
722#ifdef SIGIOT
723 /* This is missing on some systems - OS/2, for example. */
f927c5ae 724 signal (SIGIOT, fatal_error_signal);
a90538cb 725#endif
f927c5ae
JB
726#ifdef SIGEMT
727 signal (SIGEMT, fatal_error_signal);
728#endif
729 signal (SIGFPE, fatal_error_signal);
00eaaa32 730#ifdef SIGBUS
f927c5ae 731 signal (SIGBUS, fatal_error_signal);
00eaaa32 732#endif
f927c5ae 733 signal (SIGSEGV, fatal_error_signal);
00eaaa32 734#ifdef SIGSYS
f927c5ae 735 signal (SIGSYS, fatal_error_signal);
00eaaa32 736#endif
f927c5ae
JB
737 signal (SIGTERM, fatal_error_signal);
738#ifdef SIGXCPU
739 signal (SIGXCPU, fatal_error_signal);
740#endif
741#ifdef SIGXFSZ
742 signal (SIGXFSZ, fatal_error_signal);
743#endif /* SIGXFSZ */
744
271c7b7c
RS
745#ifdef SIGDANGER
746 /* This just means available memory is getting low. */
747 signal (SIGDANGER, memory_warning_signal);
748#endif
749
f927c5ae 750#ifdef AIX
56e034fa
RS
751/* 20 is SIGCHLD, 21 is SIGTTIN, 22 is SIGTTOU. */
752 signal (SIGXCPU, fatal_error_signal);
0aef8561 753#ifndef _I386
f927c5ae 754 signal (SIGIOINT, fatal_error_signal);
0aef8561 755#endif
f927c5ae
JB
756 signal (SIGGRANT, fatal_error_signal);
757 signal (SIGRETRACT, fatal_error_signal);
758 signal (SIGSOUND, fatal_error_signal);
759 signal (SIGMSG, fatal_error_signal);
760#endif /* AIX */
761 }
762
763 noninteractive1 = noninteractive;
764
765/* Perform basic initializations (not merely interning symbols) */
766
767 if (!initialized)
768 {
769 init_alloc_once ();
770 init_obarray ();
771 init_eval_once ();
772 init_syntax_once (); /* Create standard syntax table. */
773 /* Must be done before init_buffer */
774 init_casetab_once ();
775 init_buffer_once (); /* Create buffer table and some buffers */
776 init_minibuf_once (); /* Create list of minibuffers */
777 /* Must precede init_window_once */
778 init_window_once (); /* Init the window system */
779 }
780
781 init_alloc ();
f927c5ae
JB
782 init_eval ();
783 init_data ();
7074fde6 784 running_asynch_code = 0;
0e956009 785
29b89fe0
RS
786#ifdef MSDOS
787 /* Call early 'cause init_environment needs it. */
788 init_dosfns ();
789 /* Set defaults for several environment variables. */
18198bb2
RS
790 if (initialized)
791 init_environment (argc, argv, skip_args);
792 else
d1fc6752 793 tzset ();
18198bb2 794#endif /* MSDOS */
29b89fe0 795
8ba50e1a
GV
796#ifdef WINDOWSNT
797 /* Initialize environment from registry settings. */
798 init_environment ();
41f339d4 799 init_ntproc (); /* must precede init_editfns */
8ba50e1a
GV
800#endif
801
0e956009
JB
802 /* egetenv is a pretty low-level facility, which may get called in
803 many circumstances; it seems flimsy to put off initializing it
804 until calling init_callproc. */
805 set_process_environment ();
93aed04d
RS
806 /* AIX crashes are reported in system versions 3.2.3 and 3.2.4
807 if this is not done. Do it after set_process_environment so that we
808 don't pollute Vprocess_environment. */
809#ifdef AIX
810 putenv ("LANG=C");
811#endif
0e956009 812
ace40a69
RS
813 init_buffer (); /* Init default directory of main buffer */
814
7928f0b5 815 init_callproc_1 (); /* Must precede init_cmdargs and init_sys_modes. */
ace40a69 816 init_cmdargs (argc, argv, skip_args); /* Must precede init_lread. */
7928f0b5 817 init_callproc (); /* Must follow init_cmdargs but not init_sys_modes. */
fb8e9847 818 init_lread ();
f927c5ae 819
f927c5ae
JB
820 if (!noninteractive)
821 {
822#ifdef VMS
1cbd5d9d 823 init_vms_input ();/* init_display calls get_frame_size, that needs this */
f927c5ae
JB
824#endif /* VMS */
825 init_display (); /* Determine terminal type. init_sys_modes uses results */
826 }
827 init_keyboard (); /* This too must precede init_sys_modes */
f927c5ae
JB
828#ifdef VMS
829 init_vmsproc (); /* And this too. */
830#endif /* VMS */
831 init_sys_modes (); /* Init system terminal modes (RAW or CBREAK, etc.) */
832 init_xdisp ();
833 init_macros ();
834 init_editfns ();
835#ifdef LISP_FLOAT_TYPE
836 init_floatfns ();
837#endif
838#ifdef VMS
839 init_vmsfns ();
840#endif /* VMS */
f927c5ae 841 init_process ();
32676c08
JB
842#ifdef CLASH_DETECTION
843 init_filelock ();
844#endif /* CLASH_DETECTION */
f927c5ae
JB
845
846/* Intern the names of all standard functions and variables; define standard keys */
847
848 if (!initialized)
849 {
850 /* The basic levels of Lisp must come first */
851 /* And data must come first of all
852 for the sake of symbols like error-message */
853 syms_of_data ();
854 syms_of_alloc ();
fb8e9847 855 syms_of_lread ();
f927c5ae
JB
856 syms_of_print ();
857 syms_of_eval ();
858 syms_of_fns ();
f927c5ae 859 syms_of_floatfns ();
f927c5ae
JB
860
861 syms_of_abbrev ();
862 syms_of_buffer ();
863 syms_of_bytecode ();
864 syms_of_callint ();
865 syms_of_casefiddle ();
866 syms_of_casetab ();
867 syms_of_callproc ();
868 syms_of_cmds ();
869#ifndef NO_DIR_LIBRARY
870 syms_of_dired ();
871#endif /* not NO_DIR_LIBRARY */
872 syms_of_display ();
873 syms_of_doc ();
874 syms_of_editfns ();
875 syms_of_emacs ();
876 syms_of_fileio ();
877#ifdef CLASH_DETECTION
878 syms_of_filelock ();
879#endif /* CLASH_DETECTION */
880 syms_of_indent ();
881 syms_of_keyboard ();
882 syms_of_keymap ();
883 syms_of_macros ();
884 syms_of_marker ();
885 syms_of_minibuf ();
886 syms_of_mocklisp ();
f927c5ae 887 syms_of_process ();
f927c5ae 888 syms_of_search ();
1cbd5d9d 889 syms_of_frame ();
f927c5ae 890 syms_of_syntax ();
0d934e7b 891 syms_of_term ();
f927c5ae 892 syms_of_undo ();
bef79ee4
JA
893
894 /* Only defined if Emacs is compiled with USE_TEXT_PROPERTIES */
895 syms_of_textprop ();
f927c5ae
JB
896#ifdef VMS
897 syms_of_vmsproc ();
898#endif /* VMS */
05687c54
RS
899#ifdef WINDOWSNT
900 syms_of_ntproc ();
901#endif /* WINDOWSNT */
f927c5ae
JB
902 syms_of_window ();
903 syms_of_xdisp ();
904#ifdef HAVE_X_WINDOWS
72412588 905 syms_of_xterm ();
f927c5ae 906 syms_of_xfns ();
9c3f23b7 907 syms_of_xfaces ();
72412588
JB
908#ifdef HAVE_X11
909 syms_of_xselect ();
910#endif
f927c5ae
JB
911#endif /* HAVE_X_WINDOWS */
912
87485d6f
MW
913#if defined (MSDOS) && !defined (HAVE_X_WINDOWS)
914 syms_of_xfaces ();
87485d6f
MW
915#endif
916
1e9c210b 917#ifndef HAVE_NTGUI
6c850f3c 918 syms_of_xmenu ();
1e9c210b 919#endif
6c850f3c 920
8ba50e1a
GV
921#ifdef HAVE_NTGUI
922 syms_of_win32term ();
923 syms_of_win32fns ();
924 syms_of_win32faces ();
925 syms_of_win32select ();
926 syms_of_win32menu ();
927#endif /* HAVE_NTGUI */
928
f927c5ae
JB
929#ifdef SYMS_SYSTEM
930 SYMS_SYSTEM;
931#endif
932
933#ifdef SYMS_MACHINE
934 SYMS_MACHINE;
935#endif
936
937 keys_of_casefiddle ();
938 keys_of_cmds ();
939 keys_of_buffer ();
940 keys_of_keyboard ();
941 keys_of_keymap ();
942 keys_of_macros ();
943 keys_of_minibuf ();
944 keys_of_window ();
5e67fbc2 945 keys_of_frame ();
f927c5ae
JB
946 }
947
948 if (!initialized)
949 {
e2925360 950 char *file;
f927c5ae 951 /* Handle -l loadup-and-dump, args passed by Makefile. */
df6530f8 952 if (argmatch (argv, argc, "-l", "--load", 3, &file, &skip_args))
f927c5ae 953 Vtop_level = Fcons (intern ("load"),
e2925360 954 Fcons (build_string (file), Qnil));
f927c5ae
JB
955#ifdef CANNOT_DUMP
956 /* Unless next switch is -nl, load "loadup.el" first thing. */
df6530f8 957 if (!argmatch (argv, argc, "-nl", "--no-loadup", 6, NULL, &skip_args))
f927c5ae
JB
958 Vtop_level = Fcons (intern ("load"),
959 Fcons (build_string ("loadup.el"), Qnil));
960#endif /* CANNOT_DUMP */
961 }
962
93572b43
KH
963 if (initialized)
964 {
965 /* Erase any pre-dump messages in the message log, to avoid confusion */
966 Lisp_Object old_log_max;
967 old_log_max = Vmessage_log_max;
968 XSETFASTINT (Vmessage_log_max, 0);
969 message_dolog ("", 0, 1);
970 Vmessage_log_max = old_log_max;
0269dedb
RS
971
972#ifdef HAVE_TZSET
973 {
974 /* If the execution TZ happens to be the same as the dump TZ,
975 change it to some other value and then change it back,
976 to force the underlying implementation to reload the TZ info.
977 This is needed on implementations that load TZ info from files,
978 since the TZ file contents may differ between dump and execution. */
979 char *tz = getenv ("TZ");
980 if (tz && !strcmp (tz, dump_tz))
981 {
982 ++*tz;
983 tzset ();
984 --*tz;
985 }
986 }
987#endif
93572b43
KH
988 }
989
f927c5ae
JB
990 initialized = 1;
991
e7536cff
RS
992#ifdef LOCALTIME_CACHE
993 /* Some versions of localtime have a bug. They cache the value of the time
279cc2b8
JB
994 zone rather than looking it up every time. Since localtime() is
995 called to bolt the undumping time into the undumped emacs, this
afe9fae9
RS
996 results in localtime ignoring the TZ environment variable.
997 This flushes the new TZ value into localtime. */
998 tzset ();
e7536cff 999#endif /* defined (LOCALTIME_CACHE) */
279cc2b8 1000
f927c5ae
JB
1001 /* Enter editor command loop. This never returns. */
1002 Frecursive_edit ();
1003 /* NOTREACHED */
1004}
1005\f
081bef73
RS
1006/* Sort the args so we can find the most important ones
1007 at the beginning of argv. */
1008
1009/* First, here's a table of all the standard options. */
1010
1011struct standard_args
1012{
1013 char *name;
1014 char *longname;
1015 int priority;
1016 int nargs;
1017};
1018
1019struct standard_args standard_args[] =
1020{
221cbd7d
RS
1021 { "-version", "--version", 110, 0 },
1022 { "-help", "--help", 110, 0 },
081bef73 1023 { "-nl", "--no-shared-memory", 100, 0 },
2725719a 1024#ifdef VMS
081bef73 1025 { "-map", "--map-data", 100, 0 },
2725719a 1026#endif
081bef73
RS
1027 { "-t", "--terminal", 90, 1 },
1028 { "-d", "--display", 80, 1 },
1029 { "-display", 0, 80, 1 },
1030 { "-nw", "--no-windows", 70, 0 },
1031 { "-batch", "--batch", 60, 0 },
1032 { "-q", "--no-init-file", 50, 0 },
1033 { "-no-init-file", 0, 50, 0 },
1034 { "-no-site-file", "--no-site-file", 40, 0 },
1035 { "-u", "--user", 30, 1 },
1036 { "-user", 0, 30, 1 },
1037 { "-debug-init", "--debug-init", 20, 0 },
adab4483
KH
1038 { "-i", "--icon-type", 15, 0 },
1039 { "-itype", 0, 15, 0 },
f2bc3538 1040 { "-iconic", "--iconic", 15, 0 },
081bef73
RS
1041 { "-bg", "--background-color", 10, 1 },
1042 { "-background", 0, 10, 1 },
1043 { "-fg", "--foreground-color", 10, 1 },
1044 { "-foreground", 0, 10, 1 },
1045 { "-bd", "--border-color", 10, 1 },
1046 { "-bw", "--border-width", 10, 1 },
1047 { "-ib", "--internal-border", 10, 1 },
1048 { "-ms", "--mouse-color", 10, 1 },
1049 { "-cr", "--cursor-color", 10, 1 },
1050 { "-fn", "--font", 10, 1 },
1051 { "-font", 0, 10, 1 },
1052 { "-g", "--geometry", 10, 1 },
1053 { "-geometry", 0, 10, 1 },
1054 { "-T", "--title", 10, 1 },
ae63ae52 1055 { "-title", 0, 10, 1 },
081bef73
RS
1056 { "-name", "--name", 10, 1 },
1057 { "-xrm", "--xrm", 10, 1 },
fcdeb5d9
RS
1058 { "-r", "--reverse-video", 5, 0 },
1059 { "-rv", 0, 5, 0 },
1060 { "-reverse", 0, 5, 0 },
ae63ae52 1061 { "-hb", "--horizontal-scroll-bars", 5, 0 },
fcdeb5d9 1062 { "-vb", "--vertical-scroll-bars", 5, 0 },
fcdeb5d9
RS
1063 /* These have the same priority as ordinary file name args,
1064 so they are not reordered with respect to those. */
4af9e0b3
RS
1065 { "-L", "--directory", 0, 1 },
1066 { "-directory", 0, 0, 1 },
fcdeb5d9
RS
1067 { "-l", "--load", 0, 1 },
1068 { "-load", 0, 0, 1 },
1069 { "-f", "--funcall", 0, 1 },
1070 { "-funcall", 0, 0, 1 },
575985b1 1071 { "-eval", "--eval", 0, 1 },
fcdeb5d9 1072 { "-insert", "--insert", 0, 1 },
f2bc3538 1073 /* This should be processed after ordinary file name args and the like. */
fcdeb5d9 1074 { "-kill", "--kill", -10, 0 },
081bef73
RS
1075};
1076
1077/* Reorder the elements of ARGV (assumed to have ARGC elements)
1078 so that the highest priority ones come first.
1079 Do not change the order of elements of equal priority.
1080 If an option takes an argument, keep it and its argument together. */
1081
1082static void
1083sort_args (argc, argv)
1084 int argc;
1085 char **argv;
1086{
1087 char **new = (char **) xmalloc (sizeof (char *) * argc);
1088 /* For each element of argv,
1089 the corresponding element of options is:
1090 0 for an option that takes no arguments,
1091 1 for an option that takes one argument, etc.
1092 -1 for an ordinary non-option argument. */
6dad9359 1093 int *options = (int *) xmalloc (sizeof (int) * argc);
081bef73
RS
1094 int *priority = (int *) xmalloc (sizeof (int) * argc);
1095 int to = 1;
1096 int from;
1097 int i;
1098
1099 /* Categorize all the options,
1100 and figure out which argv elts are option arguments. */
1101 for (from = 1; from < argc; from++)
1102 {
1103 options[from] = -1;
fcdeb5d9 1104 priority[from] = 0;
081bef73
RS
1105 if (argv[from][0] == '-')
1106 {
1107 int match, thislen;
1108 char *equals;
1109
1110 /* Look for a match with a known old-fashioned option. */
1111 for (i = 0; i < sizeof (standard_args) / sizeof (standard_args[0]); i++)
1112 if (!strcmp (argv[from], standard_args[i].name))
1113 {
1114 options[from] = standard_args[i].nargs;
1115 priority[from] = standard_args[i].priority;
fd76ec52
RS
1116 if (from + standard_args[i].nargs >= argc)
1117 fatal ("Option `%s' requires an argument\n", argv[from]);
081bef73
RS
1118 from += standard_args[i].nargs;
1119 goto done;
1120 }
1121
1122 /* Look for a match with a known long option.
1123 MATCH is -1 if no match so far, -2 if two or more matches so far,
1124 >= 0 (the table index of the match) if just one match so far. */
1125 if (argv[from][1] == '-')
1126 {
1127 match = -1;
1128 thislen = strlen (argv[from]);
1129 equals = index (argv[from], '=');
1130 if (equals != 0)
1131 thislen = equals - argv[from];
1132
f609ef57
KH
1133 for (i = 0;
1134 i < sizeof (standard_args) / sizeof (standard_args[0]); i++)
1135 if (standard_args[i].longname
1136 && !strncmp (argv[from], standard_args[i].longname,
1137 thislen))
081bef73
RS
1138 {
1139 if (match == -1)
1140 match = i;
1141 else
1142 match = -2;
1143 }
1144
1145 /* If we found exactly one match, use that. */
1146 if (match >= 0)
1147 {
1148 options[from] = standard_args[match].nargs;
1149 priority[from] = standard_args[match].priority;
1150 /* If --OPTION=VALUE syntax is used,
1151 this option uses just one argv element. */
1152 if (equals != 0)
1153 options[from] = 0;
fd76ec52
RS
1154 if (from + options[from] >= argc)
1155 fatal ("Option `%s' requires an argument\n", argv[from]);
081bef73
RS
1156 from += options[from];
1157 }
1158 }
1159 done: ;
1160 }
1161 }
1162
1163 /* Copy the arguments, in order of decreasing priority, to NEW. */
1164 new[0] = argv[0];
1165 while (to < argc)
1166 {
1167 int best = -1;
2c70c992 1168 int best_priority = -9999;
081bef73
RS
1169
1170 /* Find the highest priority remaining option.
1171 If several have equal priority, take the first of them. */
1172 for (from = 1; from < argc; from++)
1173 {
1174 if (argv[from] != 0 && priority[from] > best_priority)
1175 {
1176 best_priority = priority[from];
1177 best = from;
1178 }
1179 /* Skip option arguments--they are tied to the options. */
1180 if (options[from] > 0)
1181 from += options[from];
1182 }
1183
1184 if (best < 0)
1185 abort ();
1186
1187 /* Copy the highest priority remaining option, with its args, to NEW. */
1188 new[to++] = argv[best];
1189 for (i = 0; i < options[best]; i++)
1190 new[to++] = argv[best + i + 1];
1191
1192 /* Clear out this option in ARGV. */
1193 argv[best] = 0;
1194 for (i = 0; i < options[best]; i++)
1195 argv[best + i + 1] = 0;
1196 }
1197
6dad9359 1198 bcopy (new, argv, sizeof (char *) * argc);
081bef73
RS
1199}
1200\f
f927c5ae 1201DEFUN ("kill-emacs", Fkill_emacs, Skill_emacs, 0, 1, "P",
edc8ae07 1202 "Exit the Emacs job and kill it.\n\
f927c5ae
JB
1203If ARG is an integer, return ARG as the exit program code.\n\
1204If ARG is a string, stuff it as keyboard input.\n\n\
1205The value of `kill-emacs-hook', if not void,\n\
1206is a list of functions (of no args),\n\
1207all of which are called before Emacs is actually killed.")
1208 (arg)
1209 Lisp_Object arg;
1210{
1211 Lisp_Object hook, hook1;
1212 int i;
1213 struct gcpro gcpro1;
1214
1215 GCPRO1 (arg);
1216
1217 if (feof (stdin))
1218 arg = Qt;
1219
2447c626 1220 if (!NILP (Vrun_hooks) && !noninteractive)
f927c5ae
JB
1221 call1 (Vrun_hooks, intern ("kill-emacs-hook"));
1222
f927c5ae
JB
1223 UNGCPRO;
1224
1225/* Is it really necessary to do this deassign
1226 when we are going to exit anyway? */
1227/* #ifdef VMS
1228 stop_vms_input ();
1229 #endif */
40be253a 1230
d0068e25 1231 shut_down_emacs (0, 0, STRINGP (arg) ? arg : Qnil);
40be253a 1232
58545838
KH
1233 /* If we have an auto-save list file,
1234 kill it because we are exiting Emacs deliberately (not crashing).
1235 Do it after shut_down_emacs, which does an auto-save. */
1236 if (STRINGP (Vauto_save_list_file_name))
1237 unlink (XSTRING (Vauto_save_list_file_name)->data);
1238
55ccc0b3 1239 exit (INTEGERP (arg) ? XINT (arg)
f927c5ae
JB
1240#ifdef VMS
1241 : 1
1242#else
1243 : 0
1244#endif
1245 );
1246 /* NOTREACHED */
1247}
40be253a
JB
1248
1249
1250/* Perform an orderly shutdown of Emacs. Autosave any modified
1251 buffers, kill any child processes, clean up the terminal modes (if
1252 we're in the foreground), and other stuff like that. Don't perform
1253 any redisplay; this may be called when Emacs is shutting down in
1254 the background, or after its X connection has died.
1255
1256 If SIG is a signal number, print a message for it.
1257
1258 This is called by fatal signal handlers, X protocol error handlers,
1259 and Fkill_emacs. */
f7ab4e3d 1260
40be253a 1261void
f7ab4e3d 1262shut_down_emacs (sig, no_x, stuff)
41423a80 1263 int sig, no_x;
f7ab4e3d 1264 Lisp_Object stuff;
40be253a 1265{
829d872b
RS
1266 /* Prevent running of hooks from now on. */
1267 Vrun_hooks = Qnil;
1268
40be253a
JB
1269 /* If we are controlling the terminal, reset terminal modes */
1270#ifdef EMACS_HAVE_TTY_PGRP
1271 {
d04d81d2
RS
1272 int pgrp = EMACS_GETPGRP (0);
1273
40be253a
JB
1274 int tpgrp;
1275 if (EMACS_GET_TTY_PGRP (0, &tpgrp) != -1
5a570e37 1276 && tpgrp == pgrp)
40be253a
JB
1277 {
1278 fflush (stdout);
1279 reset_sys_modes ();
1280 if (sig && sig != SIGTERM)
1281 fprintf (stderr, "Fatal error (%d).", sig);
1282 }
1283 }
1284#else
1285 fflush (stdout);
1286 reset_sys_modes ();
1287#endif
1288
f7ab4e3d
RS
1289 stuff_buffered_input (stuff);
1290
40be253a
JB
1291 kill_buffer_processes (Qnil);
1292 Fdo_auto_save (Qt, Qnil);
1293
1294#ifdef CLASH_DETECTION
1295 unlock_all_files ();
1296#endif
1297
1298#ifdef VMS
1299 kill_vms_processes ();
1300#endif
1301
5e7f8733 1302#if 0 /* This triggers a bug in XCloseDisplay and is not needed. */
41423a80 1303#ifdef HAVE_X_WINDOWS
f7511647
RS
1304 /* It's not safe to call intern here. Maybe we are crashing. */
1305 if (!noninteractive && SYMBOLP (Vwindow_system)
1306 && XSYMBOL (Vwindow_system)->name->size == 1
1307 && XSYMBOL (Vwindow_system)->name->data[0] == 'x'
1308 && ! no_x)
41423a80
RS
1309 Fx_close_current_connection ();
1310#endif /* HAVE_X_WINDOWS */
5e7f8733 1311#endif
41423a80 1312
40be253a
JB
1313#ifdef SIGIO
1314 /* There is a tendency for a SIGIO signal to arrive within exit,
1315 and cause a SIGHUP because the input descriptor is already closed. */
1316 unrequest_sigio ();
1317 signal (SIGIO, SIG_IGN);
1318#endif
41f339d4
RS
1319
1320#ifdef WINDOWSNT
1321 term_ntproc ();
1322#endif
40be253a
JB
1323}
1324
1325
f927c5ae
JB
1326\f
1327#ifndef CANNOT_DUMP
f927c5ae
JB
1328
1329#ifdef HAVE_SHM
1330
1331DEFUN ("dump-emacs-data", Fdump_emacs_data, Sdump_emacs_data, 1, 1, 0,
1332 "Dump current state of Emacs into data file FILENAME.\n\
1333This function exists on systems that use HAVE_SHM.")
c9aae259
EN
1334 (filename)
1335 Lisp_Object filename;
f927c5ae 1336{
55697f5b 1337 extern char my_edata[];
f927c5ae 1338 Lisp_Object tem;
f927c5ae 1339
c9aae259
EN
1340 CHECK_STRING (filename, 0);
1341 filename = Fexpand_file_name (filename, Qnil);
f927c5ae
JB
1342
1343 tem = Vpurify_flag;
1344 Vpurify_flag = Qnil;
1345
1346 fflush (stdout);
1347 /* Tell malloc where start of impure now is */
1348 /* Also arrange for warnings when nearly out of space. */
1349#ifndef SYSTEM_MALLOC
1090a161 1350 memory_warnings (my_edata, malloc_warning);
f927c5ae 1351#endif
c9aae259 1352 map_out_data (XSTRING (filename)->data);
f927c5ae
JB
1353
1354 Vpurify_flag = tem;
1355
1356 return Qnil;
1357}
1358
1359#else /* not HAVE_SHM */
1360
1361DEFUN ("dump-emacs", Fdump_emacs, Sdump_emacs, 2, 2, 0,
1362 "Dump current state of Emacs into executable file FILENAME.\n\
1363Take symbols from SYMFILE (presumably the file you executed to run Emacs).\n\
1364This is used in the file `loadup.el' when building Emacs.\n\
1365\n\
1366Bind `command-line-processed' to nil before dumping,\n\
1367if you want the dumped Emacs to process its command line\n\
1368and announce itself normally when it is run.")
c9aae259
EN
1369 (filename, symfile)
1370 Lisp_Object filename, symfile;
f927c5ae 1371{
55697f5b 1372 extern char my_edata[];
f927c5ae 1373 Lisp_Object tem;
f927c5ae 1374
c9aae259
EN
1375 CHECK_STRING (filename, 0);
1376 filename = Fexpand_file_name (filename, Qnil);
1377 if (!NILP (symfile))
f927c5ae 1378 {
c9aae259
EN
1379 CHECK_STRING (symfile, 0);
1380 if (XSTRING (symfile)->size)
1381 symfile = Fexpand_file_name (symfile, Qnil);
f927c5ae
JB
1382 }
1383
1384 tem = Vpurify_flag;
1385 Vpurify_flag = Qnil;
1386
0269dedb
RS
1387#ifdef HAVE_TZSET
1388 set_time_zone_rule (dump_tz);
1389#ifndef LOCALTIME_CACHE
1390 /* Force a tz reload, since set_time_zone_rule doesn't. */
1391 tzset ();
1392#endif
1393#endif
1394
f927c5ae
JB
1395 fflush (stdout);
1396#ifdef VMS
c9aae259 1397 mapout_data (XSTRING (filename)->data);
f927c5ae
JB
1398#else
1399 /* Tell malloc where start of impure now is */
1400 /* Also arrange for warnings when nearly out of space. */
1401#ifndef SYSTEM_MALLOC
cc5f52cb
RS
1402#ifndef WINDOWSNT
1403 /* On Windows, this was done before dumping, and that once suffices.
1404 Meanwhile, my_edata is not valid on Windows. */
cb37cf78 1405 memory_warnings (my_edata, malloc_warning);
cc5f52cb 1406#endif /* not WINDOWSNT */
f927c5ae 1407#endif
c9aae259
EN
1408 unexec (XSTRING (filename)->data,
1409 !NILP (symfile) ? XSTRING (symfile)->data : 0, my_edata, 0, 0);
f927c5ae
JB
1410#endif /* not VMS */
1411
1412 Vpurify_flag = tem;
1413
1414 return Qnil;
1415}
1416
1417#endif /* not HAVE_SHM */
1418
1419#endif /* not CANNOT_DUMP */
1420\f
4b163808 1421#ifndef SEPCHAR
f927c5ae
JB
1422#define SEPCHAR ':'
1423#endif
1424
1425Lisp_Object
1426decode_env_path (evarname, defalt)
1427 char *evarname, *defalt;
1428{
1429 register char *path, *p;
f927c5ae
JB
1430
1431 Lisp_Object lpath;
1432
2447c626
JB
1433 /* It's okay to use getenv here, because this function is only used
1434 to initialize variables when Emacs starts up, and isn't called
1435 after that. */
e065a56e
JB
1436 if (evarname != 0)
1437 path = (char *) getenv (evarname);
1438 else
1439 path = 0;
f927c5ae
JB
1440 if (!path)
1441 path = defalt;
1442 lpath = Qnil;
1443 while (1)
1444 {
1445 p = index (path, SEPCHAR);
1446 if (!p) p = path + strlen (path);
d70fbcca
RS
1447 lpath = Fcons (p - path ? make_string (path, p - path)
1448 : build_string ("."),
f927c5ae
JB
1449 lpath);
1450 if (*p)
1451 path = p + 1;
1452 else
1453 break;
1454 }
1455 return Fnreverse (lpath);
1456}
1457
1458syms_of_emacs ()
1459{
83591e66 1460#ifndef CANNOT_DUMP
f927c5ae
JB
1461#ifdef HAVE_SHM
1462 defsubr (&Sdump_emacs_data);
1463#else
1464 defsubr (&Sdump_emacs);
83591e66 1465#endif
f927c5ae
JB
1466#endif
1467
1468 defsubr (&Skill_emacs);
1469
59653951 1470 defsubr (&Sinvocation_name);
ace40a69 1471 defsubr (&Sinvocation_directory);
59653951 1472
f927c5ae
JB
1473 DEFVAR_LISP ("command-line-args", &Vcommand_line_args,
1474 "Args passed by shell to Emacs, as a list of strings.");
1475
1476 DEFVAR_LISP ("system-type", &Vsystem_type,
1477 "Value is symbol indicating type of operating system you are using.");
1478 Vsystem_type = intern (SYSTEM_TYPE);
1479
271c7b7c
RS
1480 DEFVAR_LISP ("system-configuration", &Vsystem_configuration,
1481 "Value is string indicating configuration Emacs was built for.");
f7511647 1482 Vsystem_configuration = build_string (EMACS_CONFIGURATION);
271c7b7c 1483
f0fc0b1a
KH
1484 DEFVAR_LISP ("system-configuration-options", &Vsystem_configuration_options,
1485 "String containing the configuration options Emacs was built with.");
1486 Vsystem_configuration_options = build_string (EMACS_CONFIG_OPTIONS);
1487
f927c5ae
JB
1488 DEFVAR_BOOL ("noninteractive", &noninteractive1,
1489 "Non-nil means Emacs is running without interactive terminal.");
e5d77022 1490
e5d77022 1491 DEFVAR_LISP ("kill-emacs-hook", &Vkill_emacs_hook,
edc8ae07
JB
1492 "Hook to be run whenever kill-emacs is called.\n\
1493Since kill-emacs may be invoked when the terminal is disconnected (or\n\
1494in other similar situations), functions placed on this hook should not\n\
9f232a5d
RS
1495expect to be able to interact with the user. To ask for confirmation,\n\
1496see `kill-emacs-query-functions' instead.");
edc8ae07 1497 Vkill_emacs_hook = Qnil;
3005da00
RS
1498
1499 DEFVAR_INT ("emacs-priority", &emacs_priority,
1500 "Priority for Emacs to run at.\n\
1501This value is effective only if set before Emacs is dumped,\n\
1502and only if the Emacs executable is installed with setuid to permit\n\
621ecf99 1503it to change priority. (Emacs sets its uid back to the real uid.)\n\
a422068f 1504Currently, you need to define SET_EMACS_PRIORITY in `config.h'\n\
ce305b11 1505before you compile Emacs, to enable the code for this feature.");
3005da00 1506 emacs_priority = 0;
59653951 1507
f67de86f
RS
1508 DEFVAR_LISP ("invocation-name", &Vinvocation_name,
1509 "The program name that was used to run Emacs.\n\
1510Any directory names are omitted.");
1511
1512 DEFVAR_LISP ("invocation-directory", &Vinvocation_directory,
1513 "The directory in which the Emacs executable was found, to run it.\n\
1514The value is nil if that directory's name is not known.");
1515
1516 DEFVAR_LISP ("installation-directory", &Vinstallation_directory,
1517 "A directory within which to look for the `lib-src' and `etc' directories.\n\
1518This is non-nil when we can't find those directories in their standard\n\
1519installed locations, but we can find them\n\
1520near where the Emacs executable was found.");
07f4d123 1521 Vinstallation_directory = Qnil;
f927c5ae 1522}