(struct x_display_info): Struct renamed from x_screen.
[bpt/emacs.git] / src / emacs.c
CommitLineData
f927c5ae 1/* Fully extensible Emacs, running on Unix, intended for GNU.
be06db9a 2 Copyright (C) 1985, 1986, 1987, 1993, 1994 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
18the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */
19
20
21#include <signal.h>
22#include <errno.h>
23
18160b98 24#include <config.h>
f927c5ae
JB
25#include <stdio.h>
26
27#include <sys/types.h>
28#include <sys/file.h>
29
30#ifdef VMS
31#include <ssdef.h>
32#endif
33
f927c5ae
JB
34#ifdef BSD
35#include <sys/ioctl.h>
36#endif
37
38#ifdef APOLLO
39#ifndef APOLLO_SR10
40#include <default_acl.h>
41#endif
42#endif
43
f927c5ae
JB
44#include "lisp.h"
45#include "commands.h"
bef79ee4 46#include "intervals.h"
f927c5ae 47
edc8ae07 48#include "systty.h"
8090eb09 49#include "syssignal.h"
6c362a8b 50#include "process.h"
a41f8bed 51
f927c5ae
JB
52#ifndef O_RDWR
53#define O_RDWR 2
54#endif
55
43165242
KH
56extern void malloc_warning ();
57extern char *index ();
58extern char *strerror ();
59
f927c5ae
JB
60/* Command line args from shell, as list of strings */
61Lisp_Object Vcommand_line_args;
62
59653951
JB
63/* The name under which Emacs was invoked, with any leading directory
64 names discarded. */
65Lisp_Object Vinvocation_name;
66
ace40a69
RS
67/* The directory name from which Emacs was invoked. */
68Lisp_Object Vinvocation_directory;
69
07f4d123
RS
70/* The directory name in which to find subdirs such as lisp and etc.
71 nil means get them only from PATH_LOADSEARCH. */
72Lisp_Object Vinstallation_directory;
73
e5d77022
JB
74/* Hook run by `kill-emacs' before it does really anything. */
75Lisp_Object Vkill_emacs_hook;
76
f927c5ae
JB
77/* Set nonzero after Emacs has started up the first time.
78 Prevents reinitialization of the Lisp world and keymaps
79 on subsequent starts. */
80int initialized;
81
271c7b7c 82/* Variable whose value is symbol giving operating system type. */
f927c5ae 83Lisp_Object Vsystem_type;
271c7b7c
RS
84
85/* Variable whose value is string giving configuration built for. */
86Lisp_Object Vsystem_configuration;
f0fc0b1a
KH
87
88/* Variable whose value is string giving configuration options,
89 for use when reporting bugs. */
90Lisp_Object Vsystem_configuration_options;
91
f927c5ae
JB
92/* If non-zero, emacs should not attempt to use an window-specific code,
93 but instead should use the virtual terminal under which it was started */
94int inhibit_window_system;
95
5aa7f46a
JB
96/* If nonzero, set Emacs to run at this priority. This is also used
97 in child_setup and sys_suspend to make sure subshells run at normal
98 priority; Those functions have their own extern declaration. */
3005da00
RS
99int emacs_priority;
100
647cec74 101#ifdef BSD_PGRPS
9ae8f997
JB
102/* See sysdep.c. */
103extern int inherited_pgroup;
104#endif
105
f927c5ae
JB
106#ifdef HAVE_X_WINDOWS
107/* If non-zero, -d was specified, meaning we're using some window system. */
108int display_arg;
109#endif
110
111/* An address near the bottom of the stack.
112 Tells GC how to save a copy of the stack. */
113char *stack_bottom;
114
115#ifdef HAVE_X_WINDOWS
116extern Lisp_Object Vwindow_system;
117#endif /* HAVE_X_WINDOWS */
118
119#ifdef USG_SHARED_LIBRARIES
120/* If nonzero, this is the place to put the end of the writable segment
121 at startup. */
122
123unsigned int bss_end = 0;
124#endif
125
126/* Nonzero means running Emacs without interactive terminal. */
127
128int noninteractive;
129
130/* Value of Lisp variable `noninteractive'.
131 Normally same as C variable `noninteractive'
132 but nothing terrible happens if user sets this one. */
133
134int noninteractive1;
e29f86e4
RS
135
136/* Save argv and argc. */
137char **initial_argv;
138int initial_argc;
f927c5ae
JB
139\f
140/* Signal code for the fatal signal that was received */
141int fatal_error_code;
142
143/* Nonzero if handling a fatal error already */
144int fatal_error_in_progress;
145
146/* Handle bus errors, illegal instruction, etc. */
2447c626 147SIGTYPE
f927c5ae
JB
148fatal_error_signal (sig)
149 int sig;
150{
f927c5ae
JB
151 fatal_error_code = sig;
152 signal (sig, SIG_DFL);
153
154 /* If fatal error occurs in code below, avoid infinite recursion. */
8090eb09
JB
155 if (! fatal_error_in_progress)
156 {
157 fatal_error_in_progress = 1;
f927c5ae 158
f7ab4e3d 159 shut_down_emacs (sig, 0, Qnil);
8090eb09 160 }
f927c5ae
JB
161
162#ifdef VMS
f927c5ae
JB
163 LIB$STOP (SS$_ABORT);
164#else
8090eb09
JB
165 /* Signal the same code; this time it will really be fatal.
166 Remember that since we're in a signal handler, the signal we're
167 going to send is probably blocked, so we have to unblock it if we
168 want to really receive it. */
29b89fe0 169#ifndef MSDOS
a90538cb 170 sigunblock (sigmask (fatal_error_code));
29b89fe0 171#endif
f927c5ae
JB
172 kill (getpid (), fatal_error_code);
173#endif /* not VMS */
174}
271c7b7c
RS
175
176#ifdef SIGDANGER
177
16c323ee 178/* Handler for SIGDANGER. */
271c7b7c
RS
179SIGTYPE
180memory_warning_signal (sig)
181 int sig;
182{
183 signal (sig, memory_warning_signal);
184
185 malloc_warning ("Operating system warns that virtual memory is running low.\n");
55796183
RS
186
187 /* It might be unsafe to call do_auto_save now. */
188 force_auto_save_soon ();
271c7b7c
RS
189}
190#endif
f927c5ae
JB
191\f
192/* Code for dealing with Lisp access to the Unix command line */
193
194static
195init_cmdargs (argc, argv, skip_args)
196 int argc;
197 char **argv;
198 int skip_args;
199{
200 register int i;
07f4d123 201 Lisp_Object name, dir;
f927c5ae 202
e29f86e4
RS
203 initial_argv = argv;
204 initial_argc = argc;
205
cb421be8 206 Vinvocation_name = Ffile_name_nondirectory (build_string (argv[0]));
ace40a69
RS
207 Vinvocation_directory = Ffile_name_directory (build_string (argv[0]));
208 /* If we got no directory in argv[0], search PATH to find where
209 Emacs actually came from. */
210 if (NILP (Vinvocation_directory))
211 {
212 Lisp_Object found;
213 int yes = openp (Vexec_path, Vinvocation_name,
6c362a8b 214 EXEC_SUFFIXES, &found, 1);
e443f843 215 if (yes == 1)
ace40a69
RS
216 Vinvocation_directory = Ffile_name_directory (found);
217 }
59653951 218
07f4d123
RS
219 Vinstallation_directory = Qnil;
220
221 if (!NILP (Vinvocation_directory))
222 {
223 dir = Vinvocation_directory;
224 name = Fexpand_file_name (Vinvocation_name, dir);
225 while (1)
226 {
f5ab9736 227 Lisp_Object tem, lib_src_exists;
07f4d123
RS
228 Lisp_Object etc_exists, info_exists;
229
f5ab9736
RS
230 /* See if dir contains subdirs for use by Emacs.
231 Check for the ones that would exist in a build directory,
232 not including lisp and info. */
233 tem = Fexpand_file_name (build_string ("lib-src"), dir);
234 lib_src_exists = Ffile_exists_p (tem);
235 if (!NILP (lib_src_exists))
07f4d123 236 {
f5ab9736
RS
237 tem = Fexpand_file_name (build_string ("etc"), dir);
238 etc_exists = Ffile_exists_p (tem);
239 if (!NILP (etc_exists))
07f4d123 240 {
f5ab9736
RS
241 Vinstallation_directory
242 = Ffile_name_as_directory (dir);
243 break;
07f4d123
RS
244 }
245 }
246
247 /* See if dir's parent contains those subdirs. */
f5ab9736
RS
248 tem = Fexpand_file_name (build_string ("../lib-src"), dir);
249 lib_src_exists = Ffile_exists_p (tem);
250 if (!NILP (lib_src_exists))
07f4d123 251 {
f5ab9736
RS
252 tem = Fexpand_file_name (build_string ("../etc"), dir);
253 etc_exists = Ffile_exists_p (tem);
254 if (!NILP (etc_exists))
07f4d123 255 {
f5ab9736
RS
256 tem = Fexpand_file_name (build_string (".."), dir);
257 Vinstallation_directory
258 = Ffile_name_as_directory (tem);
259 break;
07f4d123
RS
260 }
261 }
262
263 /* If the Emacs executable is actually a link,
264 next try the dir that the link points into. */
265 tem = Ffile_symlink_p (name);
266 if (!NILP (tem))
267 {
260ec24d 268 name = Fexpand_file_name (tem, dir);
07f4d123
RS
269 dir = Ffile_name_directory (name);
270 }
271 else
272 break;
273 }
274 }
275
f927c5ae
JB
276 Vcommand_line_args = Qnil;
277
278 for (i = argc - 1; i >= 0; i--)
279 {
280 if (i == 0 || i > skip_args)
281 Vcommand_line_args
282 = Fcons (build_string (argv[i]), Vcommand_line_args);
283 }
284}
59653951
JB
285
286DEFUN ("invocation-name", Finvocation_name, Sinvocation_name, 0, 0, 0,
287 "Return the program name that was used to run Emacs.\n\
288Any directory names are omitted.")
289 ()
290{
291 return Fcopy_sequence (Vinvocation_name);
292}
293
ace40a69
RS
294DEFUN ("invocation-directory", Finvocation_directory, Sinvocation_directory,
295 0, 0, 0,
296 "Return the directory name in which the Emacs executable was located")
297 ()
298{
299 return Fcopy_sequence (Vinvocation_directory);
300}
301
f927c5ae
JB
302\f
303#ifdef VMS
304#ifdef LINK_CRTL_SHARE
305#ifdef SHAREABLE_LIB_BUG
306extern noshare char **environ;
307#endif /* SHAREABLE_LIB_BUG */
308#endif /* LINK_CRTL_SHARE */
309#endif /* VMS */
310
a90538cb 311#ifndef ORDINARY_LINK
efd241cc
RS
312/* We don't include crtbegin.o and crtend.o in the link,
313 so these functions and variables might be missed.
314 Provide dummy definitions to avoid error.
315 (We don't have any real constructors or destructors.) */
316#ifdef __GNUC__
46e65b73 317#ifndef GCC_CTORS_IN_LIBC
c83a7064 318__do_global_ctors ()
efd241cc 319{}
c83a7064
RS
320__do_global_ctors_aux ()
321{}
33143604
RS
322__do_global_dtors ()
323{}
64c1864a
RS
324/* Linux has a bug in its library; avoid an error. */
325#ifndef LINUX
c83a7064 326char * __CTOR_LIST__[2] = { (char *) (-1), 0 };
64c1864a 327#endif
c83a7064 328char * __DTOR_LIST__[2] = { (char *) (-1), 0 };
46e65b73 329#endif /* GCC_CTORS_IN_LIBC */
c83a7064 330__main ()
efd241cc 331{}
efd241cc 332#endif /* __GNUC__ */
a90538cb 333#endif /* ORDINARY_LINK */
efd241cc 334
e2925360
KH
335/* Test whether the next argument in ARGV matches SSTR or a prefix of
336 LSTR (at least MINLEN characters). If so, then if VALPTR is non-null
337 (the argument is supposed to have a value) store in *VALPTR either
338 the next argument or the portion of this one after the equal sign.
339 ARGV is read starting at position *SKIPPTR; this index is advanced
340 by the number of arguments used.
341
342 Too bad we can't just use getopt for all of this, but we don't have
343 enough information to do it right. */
344static int
345argmatch (argv, sstr, lstr, minlen, valptr, skipptr)
346 char **argv;
347 char *sstr;
348 char *lstr;
349 int minlen;
350 char **valptr;
351 int *skipptr;
352{
353 char *p;
354 int arglen;
355 char *arg = argv[*skipptr+1];
356 if (arg == NULL)
357 return 0;
358 if (strcmp (arg, sstr) == 0)
359 {
360 if (valptr != NULL)
361 {
362 *valptr = argv[*skipptr+2];
363 *skipptr += 2;
364 }
365 else
366 *skipptr += 1;
367 return 1;
368 }
369 arglen = (valptr != NULL && (p = index (arg, '=')) != NULL
370 ? p - arg : strlen (arg));
371 if (arglen < minlen || strncmp (arg, lstr, arglen) != 0)
372 return 0;
373 else if (valptr == NULL)
374 {
375 *skipptr += 1;
376 return 1;
377 }
378 else if (p != NULL)
379 {
380 *valptr = p+1;
381 *skipptr += 1;
382 return 1;
383 }
384 else if (argv[*skipptr+2] != NULL)
385 {
386 *valptr = argv[*skipptr+2];
387 *skipptr += 2;
388 return 1;
389 }
390 else
391 {
392 return 0;
393 }
394}
395
f927c5ae
JB
396/* ARGSUSED */
397main (argc, argv, envp)
398 int argc;
399 char **argv;
400 char **envp;
401{
402 char stack_bottom_variable;
403 int skip_args = 0;
404 extern int errno;
405 extern sys_nerr;
f927c5ae
JB
406
407/* Map in shared memory, if we are using that. */
408#ifdef HAVE_SHM
e2925360 409 if (argmatch (argv, "-nl", "--no-shared-memory", 6, NULL, &skip_args))
f927c5ae
JB
410 {
411 map_in_data (0);
412 /* The shared memory was just restored, which clobbered this. */
413 skip_args = 1;
414 }
415 else
416 {
417 map_in_data (1);
418 /* The shared memory was just restored, which clobbered this. */
419 skip_args = 0;
420 }
421#endif
422
19a36ec6 423#ifdef NeXT
4b163808 424 extern int malloc_cookie;
19a36ec6
RS
425
426 /* This helps out unexnext.c. */
427 if (initialized)
428 if (malloc_jumpstart (malloc_cookie) != 0)
429 printf ("malloc jumpstart failed!\n");
430#endif /* NeXT */
431
f927c5ae 432#ifdef HAVE_X_WINDOWS
fb8e9847
JB
433 /* Stupid kludge to catch command-line display spec. We can't
434 handle this argument entirely in window system dependent code
435 because we don't even know which window system dependent code
436 to run until we've recognized this argument. */
f927c5ae
JB
437 {
438 int i;
439
e2925360
KH
440 /* We don't check for a long option --display here, since the X code
441 won't be able to recognize that form anyway. */
f927c5ae 442 for (i = 1; (i < argc && ! display_arg); i++)
92381c7c 443 if (!strcmp (argv[i], "-d") || !strcmp (argv[i], "-display"))
f927c5ae
JB
444 display_arg = 1;
445 }
446#endif
447
448#ifdef VMS
449 /* If -map specified, map the data file in */
e2925360
KH
450 {
451 char *file;
452 if (argmatch (argv, "-map", "--map-data", 3, &mapin_file, &skip_args))
453 mapin_data (file);
454 }
f927c5ae
JB
455
456#ifdef LINK_CRTL_SHARE
457#ifdef SHAREABLE_LIB_BUG
458 /* Bletcherous shared libraries! */
459 if (!stdin)
460 stdin = fdopen (0, "r");
461 if (!stdout)
462 stdout = fdopen (1, "w");
463 if (!stderr)
464 stderr = fdopen (2, "w");
465 if (!environ)
466 environ = envp;
467#endif /* SHAREABLE_LIB_BUG */
468#endif /* LINK_CRTL_SHARE */
469#endif /* VMS */
470
471 /* Record (approximately) where the stack begins. */
472 stack_bottom = &stack_bottom_variable;
473
474#ifdef RUN_TIME_REMAP
475 if (initialized)
476 run_time_remap (argv[0]);
477#endif
478
479#ifdef USG_SHARED_LIBRARIES
480 if (bss_end)
481 brk (bss_end);
482#endif
483
484 clearerr (stdin);
9ae8f997 485
f927c5ae
JB
486#ifdef APOLLO
487#ifndef APOLLO_SR10
488 /* If USE_DOMAIN_ACLS environment variable exists,
489 use ACLs rather than UNIX modes. */
490 if (egetenv ("USE_DOMAIN_ACLS"))
491 default_acl (USE_DEFACL);
492#endif
493#endif /* APOLLO */
494
495#ifndef SYSTEM_MALLOC
496 if (! initialized)
9ac0d9e0
JB
497 {
498 /* Arrange to get warning messages as memory fills up. */
499 memory_warnings (0, malloc_warning);
500
501 /* Arrange to disable interrupt input while malloc and friends are
502 running. */
503 uninterrupt_malloc ();
504 }
f927c5ae
JB
505#endif /* not SYSTEM_MALLOC */
506
29b89fe0
RS
507#ifdef MSDOS
508 /* We do all file input/output as binary files. When we need to translate
509 newlines, we do that manually. */
510 _fmode = O_BINARY;
511 (stdin)->_flag &= ~_IOTEXT;
512 (stdout)->_flag &= ~_IOTEXT;
513 (stderr)->_flag &= ~_IOTEXT;
514#endif /* MSDOS */
515
a422068f 516#ifdef SET_EMACS_PRIORITY
3005da00 517 if (emacs_priority)
5aa7f46a 518 nice (emacs_priority);
f927c5ae 519 setuid (getuid ());
a422068f 520#endif /* SET_EMACS_PRIORITY */
f927c5ae 521
d8b3a65d 522#ifdef EXTRA_INITIALIZE
a9260219 523 EXTRA_INITIALIZE;
d8b3a65d
RS
524#endif
525
f927c5ae
JB
526 inhibit_window_system = 0;
527
4fc0b45b 528 /* Handle the -t switch, which specifies filename to use as terminal */
e2925360
KH
529 {
530 char *term;
531 if (argmatch (argv, "-t", "--terminal", 4, &term, &skip_args))
532 {
533 int result;
534 close (0);
535 close (1);
536 result = open (term, O_RDWR, 2 );
537 if (result < 0)
538 {
539 char *errstring = strerror (errno);
540 fprintf (stderr, "emacs: %s: %s\n", term, errstring);
541 exit (1);
542 }
543 dup (0);
544 if (! isatty (0))
545 {
546 fprintf (stderr, "emacs: %s: not a tty\n", term);
547 exit (1);
548 }
549 fprintf (stderr, "Using %s\n", term);
f927c5ae 550#ifdef HAVE_X_WINDOWS
e2925360 551 inhibit_window_system = 1; /* -t => -nw */
f927c5ae 552#endif
e2925360
KH
553 }
554 }
555 if (argmatch (argv, "-nw", "--no-windows", 6, NULL, &skip_args))
556 inhibit_window_system = 1;
f927c5ae 557
e2925360 558 /* Handle the -batch switch, which means don't do interactive display. */
f927c5ae 559 noninteractive = 0;
e2925360
KH
560 if (argmatch (argv, "-batch", "--batch", 5, NULL, &skip_args))
561 noninteractive = 1;
562
563 /* Handle the --help option, which gives a usage message.. */
564 if (argmatch (argv, "-help", "--help", 3, NULL, &skip_args))
f927c5ae 565 {
e2925360
KH
566 printf ("\
567Usage: %s [-t term] [--terminal term] [-nw] [--no-windows] [--batch]\n\
568 [-q] [--no-init-file] [-u user] [--user user] [--debug-init]\n\
569\(Arguments above this line must be first; those below may be in any order)\n\
570 [-f func] [--funcall func] [-l file] [--load file] [--insert file]\n\
571 file-to-visit [--kill]\n", argv[0]);
572 exit (0);
f927c5ae
JB
573 }
574
edb85f59
RS
575 if (! noninteractive)
576 {
577#ifdef BSD_PGRPS
578 if (initialized)
579 {
580 inherited_pgroup = EMACS_GETPGRP (0);
581 setpgrp (0, getpid ());
582 }
583#else
584#if defined (USG5) && defined (INTERRUPT_INPUT)
585 setpgrp ();
586#endif
587#endif
588 }
589
fb8e9847
JB
590#ifdef POSIX_SIGNALS
591 init_signals ();
592#endif
593
f927c5ae
JB
594 if (
595#ifndef CANNOT_DUMP
596 ! noninteractive || initialized
597#else
598 1
599#endif
600 )
601 {
602 /* Don't catch these signals in batch mode if not initialized.
603 On some machines, this sets static data that would make
604 signal fail to work right when the dumped Emacs is run. */
605 signal (SIGHUP, fatal_error_signal);
606 signal (SIGQUIT, fatal_error_signal);
607 signal (SIGILL, fatal_error_signal);
608 signal (SIGTRAP, fatal_error_signal);
a90538cb
JB
609#ifdef SIGIOT
610 /* This is missing on some systems - OS/2, for example. */
f927c5ae 611 signal (SIGIOT, fatal_error_signal);
a90538cb 612#endif
f927c5ae
JB
613#ifdef SIGEMT
614 signal (SIGEMT, fatal_error_signal);
615#endif
616 signal (SIGFPE, fatal_error_signal);
00eaaa32 617#ifdef SIGBUS
f927c5ae 618 signal (SIGBUS, fatal_error_signal);
00eaaa32 619#endif
f927c5ae 620 signal (SIGSEGV, fatal_error_signal);
00eaaa32 621#ifdef SIGSYS
f927c5ae 622 signal (SIGSYS, fatal_error_signal);
00eaaa32 623#endif
f927c5ae
JB
624 signal (SIGTERM, fatal_error_signal);
625#ifdef SIGXCPU
626 signal (SIGXCPU, fatal_error_signal);
627#endif
628#ifdef SIGXFSZ
629 signal (SIGXFSZ, fatal_error_signal);
630#endif /* SIGXFSZ */
631
271c7b7c
RS
632#ifdef SIGDANGER
633 /* This just means available memory is getting low. */
634 signal (SIGDANGER, memory_warning_signal);
635#endif
636
f927c5ae 637#ifdef AIX
56e034fa
RS
638/* 20 is SIGCHLD, 21 is SIGTTIN, 22 is SIGTTOU. */
639 signal (SIGXCPU, fatal_error_signal);
0aef8561 640#ifndef _I386
f927c5ae 641 signal (SIGIOINT, fatal_error_signal);
0aef8561 642#endif
f927c5ae
JB
643 signal (SIGGRANT, fatal_error_signal);
644 signal (SIGRETRACT, fatal_error_signal);
645 signal (SIGSOUND, fatal_error_signal);
646 signal (SIGMSG, fatal_error_signal);
647#endif /* AIX */
648 }
649
650 noninteractive1 = noninteractive;
651
652/* Perform basic initializations (not merely interning symbols) */
653
654 if (!initialized)
655 {
656 init_alloc_once ();
657 init_obarray ();
658 init_eval_once ();
659 init_syntax_once (); /* Create standard syntax table. */
660 /* Must be done before init_buffer */
661 init_casetab_once ();
662 init_buffer_once (); /* Create buffer table and some buffers */
663 init_minibuf_once (); /* Create list of minibuffers */
664 /* Must precede init_window_once */
665 init_window_once (); /* Init the window system */
666 }
667
668 init_alloc ();
f927c5ae
JB
669 init_eval ();
670 init_data ();
0e956009 671
29b89fe0
RS
672#ifdef MSDOS
673 /* Call early 'cause init_environment needs it. */
674 init_dosfns ();
675 /* Set defaults for several environment variables. */
676 if (initialized) init_environment (argc, argv, skip_args);
677#endif
678
0e956009
JB
679 /* egetenv is a pretty low-level facility, which may get called in
680 many circumstances; it seems flimsy to put off initializing it
681 until calling init_callproc. */
682 set_process_environment ();
93aed04d
RS
683 /* AIX crashes are reported in system versions 3.2.3 and 3.2.4
684 if this is not done. Do it after set_process_environment so that we
685 don't pollute Vprocess_environment. */
686#ifdef AIX
687 putenv ("LANG=C");
688#endif
0e956009 689
ace40a69
RS
690 init_buffer (); /* Init default directory of main buffer */
691
7928f0b5 692 init_callproc_1 (); /* Must precede init_cmdargs and init_sys_modes. */
ace40a69 693 init_cmdargs (argc, argv, skip_args); /* Must precede init_lread. */
7928f0b5 694 init_callproc (); /* Must follow init_cmdargs but not init_sys_modes. */
fb8e9847 695 init_lread ();
f927c5ae 696
f927c5ae
JB
697 if (!noninteractive)
698 {
699#ifdef VMS
1cbd5d9d 700 init_vms_input ();/* init_display calls get_frame_size, that needs this */
f927c5ae
JB
701#endif /* VMS */
702 init_display (); /* Determine terminal type. init_sys_modes uses results */
703 }
704 init_keyboard (); /* This too must precede init_sys_modes */
f927c5ae
JB
705#ifdef VMS
706 init_vmsproc (); /* And this too. */
707#endif /* VMS */
708 init_sys_modes (); /* Init system terminal modes (RAW or CBREAK, etc.) */
709 init_xdisp ();
710 init_macros ();
711 init_editfns ();
712#ifdef LISP_FLOAT_TYPE
713 init_floatfns ();
714#endif
715#ifdef VMS
716 init_vmsfns ();
717#endif /* VMS */
f927c5ae 718 init_process ();
32676c08
JB
719#ifdef CLASH_DETECTION
720 init_filelock ();
721#endif /* CLASH_DETECTION */
f927c5ae
JB
722
723/* Intern the names of all standard functions and variables; define standard keys */
724
725 if (!initialized)
726 {
727 /* The basic levels of Lisp must come first */
728 /* And data must come first of all
729 for the sake of symbols like error-message */
730 syms_of_data ();
731 syms_of_alloc ();
fb8e9847 732 syms_of_lread ();
f927c5ae
JB
733 syms_of_print ();
734 syms_of_eval ();
735 syms_of_fns ();
f927c5ae 736 syms_of_floatfns ();
f927c5ae
JB
737
738 syms_of_abbrev ();
739 syms_of_buffer ();
740 syms_of_bytecode ();
741 syms_of_callint ();
742 syms_of_casefiddle ();
743 syms_of_casetab ();
744 syms_of_callproc ();
745 syms_of_cmds ();
746#ifndef NO_DIR_LIBRARY
747 syms_of_dired ();
748#endif /* not NO_DIR_LIBRARY */
749 syms_of_display ();
750 syms_of_doc ();
751 syms_of_editfns ();
752 syms_of_emacs ();
753 syms_of_fileio ();
754#ifdef CLASH_DETECTION
755 syms_of_filelock ();
756#endif /* CLASH_DETECTION */
757 syms_of_indent ();
758 syms_of_keyboard ();
759 syms_of_keymap ();
760 syms_of_macros ();
761 syms_of_marker ();
762 syms_of_minibuf ();
763 syms_of_mocklisp ();
f927c5ae 764 syms_of_process ();
f927c5ae 765 syms_of_search ();
1cbd5d9d 766 syms_of_frame ();
f927c5ae 767 syms_of_syntax ();
0d934e7b 768 syms_of_term ();
f927c5ae 769 syms_of_undo ();
bef79ee4
JA
770
771 /* Only defined if Emacs is compiled with USE_TEXT_PROPERTIES */
772 syms_of_textprop ();
f927c5ae
JB
773#ifdef VMS
774 syms_of_vmsproc ();
775#endif /* VMS */
776 syms_of_window ();
777 syms_of_xdisp ();
778#ifdef HAVE_X_WINDOWS
72412588 779 syms_of_xterm ();
f927c5ae 780 syms_of_xfns ();
9c3f23b7 781 syms_of_xfaces ();
72412588
JB
782#ifdef HAVE_X11
783 syms_of_xselect ();
784#endif
dbc4e1c1 785#ifdef HAVE_X_MENU
f927c5ae
JB
786 syms_of_xmenu ();
787#endif /* HAVE_X_MENU */
788#endif /* HAVE_X_WINDOWS */
789
87485d6f
MW
790#if defined (MSDOS) && !defined (HAVE_X_WINDOWS)
791 syms_of_xfaces ();
792 syms_of_xmenu ();
793#endif
794
f927c5ae
JB
795#ifdef SYMS_SYSTEM
796 SYMS_SYSTEM;
797#endif
798
799#ifdef SYMS_MACHINE
800 SYMS_MACHINE;
801#endif
802
803 keys_of_casefiddle ();
804 keys_of_cmds ();
805 keys_of_buffer ();
806 keys_of_keyboard ();
807 keys_of_keymap ();
808 keys_of_macros ();
809 keys_of_minibuf ();
810 keys_of_window ();
5e67fbc2 811 keys_of_frame ();
f927c5ae
JB
812 }
813
814 if (!initialized)
815 {
e2925360 816 char *file;
f927c5ae 817 /* Handle -l loadup-and-dump, args passed by Makefile. */
e2925360 818 if (argmatch (argv, "-l", "--load", 3, &file, &skip_args))
f927c5ae 819 Vtop_level = Fcons (intern ("load"),
e2925360 820 Fcons (build_string (file), Qnil));
f927c5ae
JB
821#ifdef CANNOT_DUMP
822 /* Unless next switch is -nl, load "loadup.el" first thing. */
e2925360 823 if (!argmatch (argv, "-nl", "--no-loadup", 6, NULL, &skip_args))
f927c5ae
JB
824 Vtop_level = Fcons (intern ("load"),
825 Fcons (build_string ("loadup.el"), Qnil));
826#endif /* CANNOT_DUMP */
827 }
828
829 initialized = 1;
830
afe9fae9
RS
831#if defined (sun) || defined (LOCALTIME_CACHE)
832 /* sun's localtime has a bug. it caches the value of the time
279cc2b8
JB
833 zone rather than looking it up every time. Since localtime() is
834 called to bolt the undumping time into the undumped emacs, this
afe9fae9
RS
835 results in localtime ignoring the TZ environment variable.
836 This flushes the new TZ value into localtime. */
837 tzset ();
838#endif /* defined (sun) || defined (LOCALTIME_CACHE) */
279cc2b8 839
e2925360
KH
840 /* Handle the GNU standard option --version. */
841 if (argmatch (argv, "-version", "--version", 3, NULL, &skip_args))
842 {
843 Lisp_Object ver;
844 ver = call0 (intern ("emacs-version"));
845 if (STRINGP (ver))
846 printf ("%s\n", XSTRING (ver)->data);
847 exit (0);
848 }
849
f927c5ae
JB
850 /* Enter editor command loop. This never returns. */
851 Frecursive_edit ();
852 /* NOTREACHED */
853}
854\f
855DEFUN ("kill-emacs", Fkill_emacs, Skill_emacs, 0, 1, "P",
edc8ae07 856 "Exit the Emacs job and kill it.\n\
f927c5ae
JB
857If ARG is an integer, return ARG as the exit program code.\n\
858If ARG is a string, stuff it as keyboard input.\n\n\
859The value of `kill-emacs-hook', if not void,\n\
860is a list of functions (of no args),\n\
861all of which are called before Emacs is actually killed.")
862 (arg)
863 Lisp_Object arg;
864{
865 Lisp_Object hook, hook1;
866 int i;
867 struct gcpro gcpro1;
868
869 GCPRO1 (arg);
870
871 if (feof (stdin))
872 arg = Qt;
873
2447c626 874 if (!NILP (Vrun_hooks) && !noninteractive)
f927c5ae
JB
875 call1 (Vrun_hooks, intern ("kill-emacs-hook"));
876
f927c5ae
JB
877 UNGCPRO;
878
879/* Is it really necessary to do this deassign
880 when we are going to exit anyway? */
881/* #ifdef VMS
882 stop_vms_input ();
883 #endif */
40be253a 884
d0068e25 885 shut_down_emacs (0, 0, STRINGP (arg) ? arg : Qnil);
40be253a 886
55ccc0b3 887 exit (INTEGERP (arg) ? XINT (arg)
f927c5ae
JB
888#ifdef VMS
889 : 1
890#else
891 : 0
892#endif
893 );
894 /* NOTREACHED */
895}
40be253a
JB
896
897
898/* Perform an orderly shutdown of Emacs. Autosave any modified
899 buffers, kill any child processes, clean up the terminal modes (if
900 we're in the foreground), and other stuff like that. Don't perform
901 any redisplay; this may be called when Emacs is shutting down in
902 the background, or after its X connection has died.
903
904 If SIG is a signal number, print a message for it.
905
906 This is called by fatal signal handlers, X protocol error handlers,
907 and Fkill_emacs. */
f7ab4e3d 908
40be253a 909void
f7ab4e3d 910shut_down_emacs (sig, no_x, stuff)
41423a80 911 int sig, no_x;
f7ab4e3d 912 Lisp_Object stuff;
40be253a 913{
829d872b
RS
914 /* Prevent running of hooks from now on. */
915 Vrun_hooks = Qnil;
916
40be253a
JB
917 /* If we are controlling the terminal, reset terminal modes */
918#ifdef EMACS_HAVE_TTY_PGRP
919 {
d04d81d2
RS
920 int pgrp = EMACS_GETPGRP (0);
921
40be253a
JB
922 int tpgrp;
923 if (EMACS_GET_TTY_PGRP (0, &tpgrp) != -1
5a570e37 924 && tpgrp == pgrp)
40be253a
JB
925 {
926 fflush (stdout);
927 reset_sys_modes ();
928 if (sig && sig != SIGTERM)
929 fprintf (stderr, "Fatal error (%d).", sig);
930 }
931 }
932#else
933 fflush (stdout);
934 reset_sys_modes ();
935#endif
936
f7ab4e3d
RS
937 stuff_buffered_input (stuff);
938
40be253a
JB
939 kill_buffer_processes (Qnil);
940 Fdo_auto_save (Qt, Qnil);
941
942#ifdef CLASH_DETECTION
943 unlock_all_files ();
944#endif
945
946#ifdef VMS
947 kill_vms_processes ();
948#endif
949
5e7f8733 950#if 0 /* This triggers a bug in XCloseDisplay and is not needed. */
41423a80 951#ifdef HAVE_X_WINDOWS
f7511647
RS
952 /* It's not safe to call intern here. Maybe we are crashing. */
953 if (!noninteractive && SYMBOLP (Vwindow_system)
954 && XSYMBOL (Vwindow_system)->name->size == 1
955 && XSYMBOL (Vwindow_system)->name->data[0] == 'x'
956 && ! no_x)
41423a80
RS
957 Fx_close_current_connection ();
958#endif /* HAVE_X_WINDOWS */
5e7f8733 959#endif
41423a80 960
40be253a
JB
961#ifdef SIGIO
962 /* There is a tendency for a SIGIO signal to arrive within exit,
963 and cause a SIGHUP because the input descriptor is already closed. */
964 unrequest_sigio ();
965 signal (SIGIO, SIG_IGN);
966#endif
967}
968
969
f927c5ae
JB
970\f
971#ifndef CANNOT_DUMP
972/* Nothing like this can be implemented on an Apollo.
973 What a loss! */
974
975#ifdef HAVE_SHM
976
977DEFUN ("dump-emacs-data", Fdump_emacs_data, Sdump_emacs_data, 1, 1, 0,
978 "Dump current state of Emacs into data file FILENAME.\n\
979This function exists on systems that use HAVE_SHM.")
980 (intoname)
981 Lisp_Object intoname;
982{
983 extern int my_edata;
984 Lisp_Object tem;
f927c5ae
JB
985
986 CHECK_STRING (intoname, 0);
987 intoname = Fexpand_file_name (intoname, Qnil);
988
989 tem = Vpurify_flag;
990 Vpurify_flag = Qnil;
991
992 fflush (stdout);
993 /* Tell malloc where start of impure now is */
994 /* Also arrange for warnings when nearly out of space. */
995#ifndef SYSTEM_MALLOC
70eb2c3e 996 memory_warnings (&my_edata, malloc_warning);
f927c5ae
JB
997#endif
998 map_out_data (XSTRING (intoname)->data);
999
1000 Vpurify_flag = tem;
1001
1002 return Qnil;
1003}
1004
1005#else /* not HAVE_SHM */
1006
1007DEFUN ("dump-emacs", Fdump_emacs, Sdump_emacs, 2, 2, 0,
1008 "Dump current state of Emacs into executable file FILENAME.\n\
1009Take symbols from SYMFILE (presumably the file you executed to run Emacs).\n\
1010This is used in the file `loadup.el' when building Emacs.\n\
1011\n\
1012Bind `command-line-processed' to nil before dumping,\n\
1013if you want the dumped Emacs to process its command line\n\
1014and announce itself normally when it is run.")
1015 (intoname, symname)
1016 Lisp_Object intoname, symname;
1017{
1018 extern int my_edata;
1019 Lisp_Object tem;
f927c5ae
JB
1020
1021 CHECK_STRING (intoname, 0);
1022 intoname = Fexpand_file_name (intoname, Qnil);
2447c626 1023 if (!NILP (symname))
f927c5ae
JB
1024 {
1025 CHECK_STRING (symname, 0);
1026 if (XSTRING (symname)->size)
1027 symname = Fexpand_file_name (symname, Qnil);
1028 }
1029
1030 tem = Vpurify_flag;
1031 Vpurify_flag = Qnil;
1032
1033 fflush (stdout);
1034#ifdef VMS
1035 mapout_data (XSTRING (intoname)->data);
1036#else
1037 /* Tell malloc where start of impure now is */
1038 /* Also arrange for warnings when nearly out of space. */
1039#ifndef SYSTEM_MALLOC
70eb2c3e 1040 memory_warnings (&my_edata, malloc_warning);
f927c5ae
JB
1041#endif
1042 unexec (XSTRING (intoname)->data,
2447c626 1043 !NILP (symname) ? XSTRING (symname)->data : 0, &my_edata, 0, 0);
f927c5ae
JB
1044#endif /* not VMS */
1045
1046 Vpurify_flag = tem;
1047
1048 return Qnil;
1049}
1050
1051#endif /* not HAVE_SHM */
1052
1053#endif /* not CANNOT_DUMP */
1054\f
4b163808 1055#ifndef SEPCHAR
f927c5ae
JB
1056#define SEPCHAR ':'
1057#endif
1058
1059Lisp_Object
1060decode_env_path (evarname, defalt)
1061 char *evarname, *defalt;
1062{
1063 register char *path, *p;
f927c5ae
JB
1064
1065 Lisp_Object lpath;
1066
2447c626
JB
1067 /* It's okay to use getenv here, because this function is only used
1068 to initialize variables when Emacs starts up, and isn't called
1069 after that. */
e065a56e
JB
1070 if (evarname != 0)
1071 path = (char *) getenv (evarname);
1072 else
1073 path = 0;
f927c5ae
JB
1074 if (!path)
1075 path = defalt;
1076 lpath = Qnil;
1077 while (1)
1078 {
1079 p = index (path, SEPCHAR);
1080 if (!p) p = path + strlen (path);
1081 lpath = Fcons (p - path ? make_string (path, p - path) : Qnil,
1082 lpath);
1083 if (*p)
1084 path = p + 1;
1085 else
1086 break;
1087 }
1088 return Fnreverse (lpath);
1089}
1090
1091syms_of_emacs ()
1092{
83591e66 1093#ifndef CANNOT_DUMP
f927c5ae
JB
1094#ifdef HAVE_SHM
1095 defsubr (&Sdump_emacs_data);
1096#else
1097 defsubr (&Sdump_emacs);
83591e66 1098#endif
f927c5ae
JB
1099#endif
1100
1101 defsubr (&Skill_emacs);
1102
59653951 1103 defsubr (&Sinvocation_name);
ace40a69 1104 defsubr (&Sinvocation_directory);
59653951 1105
f927c5ae
JB
1106 DEFVAR_LISP ("command-line-args", &Vcommand_line_args,
1107 "Args passed by shell to Emacs, as a list of strings.");
1108
1109 DEFVAR_LISP ("system-type", &Vsystem_type,
1110 "Value is symbol indicating type of operating system you are using.");
1111 Vsystem_type = intern (SYSTEM_TYPE);
1112
271c7b7c
RS
1113 DEFVAR_LISP ("system-configuration", &Vsystem_configuration,
1114 "Value is string indicating configuration Emacs was built for.");
f7511647 1115 Vsystem_configuration = build_string (EMACS_CONFIGURATION);
271c7b7c 1116
f0fc0b1a
KH
1117 DEFVAR_LISP ("system-configuration-options", &Vsystem_configuration_options,
1118 "String containing the configuration options Emacs was built with.");
1119 Vsystem_configuration_options = build_string (EMACS_CONFIG_OPTIONS);
1120
f927c5ae
JB
1121 DEFVAR_BOOL ("noninteractive", &noninteractive1,
1122 "Non-nil means Emacs is running without interactive terminal.");
e5d77022 1123
e5d77022 1124 DEFVAR_LISP ("kill-emacs-hook", &Vkill_emacs_hook,
edc8ae07
JB
1125 "Hook to be run whenever kill-emacs is called.\n\
1126Since kill-emacs may be invoked when the terminal is disconnected (or\n\
1127in other similar situations), functions placed on this hook should not\n\
679aee02 1128expect to be able to interact with the user.");
edc8ae07 1129 Vkill_emacs_hook = Qnil;
3005da00
RS
1130
1131 DEFVAR_INT ("emacs-priority", &emacs_priority,
1132 "Priority for Emacs to run at.\n\
1133This value is effective only if set before Emacs is dumped,\n\
1134and only if the Emacs executable is installed with setuid to permit\n\
621ecf99 1135it to change priority. (Emacs sets its uid back to the real uid.)\n\
a422068f 1136Currently, you need to define SET_EMACS_PRIORITY in `config.h'\n\
ce305b11 1137before you compile Emacs, to enable the code for this feature.");
3005da00 1138 emacs_priority = 0;
59653951 1139
f67de86f
RS
1140 DEFVAR_LISP ("invocation-name", &Vinvocation_name,
1141 "The program name that was used to run Emacs.\n\
1142Any directory names are omitted.");
1143
1144 DEFVAR_LISP ("invocation-directory", &Vinvocation_directory,
1145 "The directory in which the Emacs executable was found, to run it.\n\
1146The value is nil if that directory's name is not known.");
1147
1148 DEFVAR_LISP ("installation-directory", &Vinstallation_directory,
1149 "A directory within which to look for the `lib-src' and `etc' directories.\n\
1150This is non-nil when we can't find those directories in their standard\n\
1151installed locations, but we can find them\n\
1152near where the Emacs executable was found.");
07f4d123 1153 Vinstallation_directory = Qnil;
f927c5ae 1154}