]> code.delx.au - gnu-emacs/blob - src/emacs.c
(main): Use Vbuffer_alist instead of Fbuffer_list.
[gnu-emacs] / src / emacs.c
1 /* Fully extensible Emacs, running on Unix, intended for GNU.
2 Copyright (C) 1985,86,87,93,94,95,97,1998 Free Software Foundation, Inc.
3
4 This file is part of GNU Emacs.
5
6 GNU Emacs is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2, or (at your option)
9 any later version.
10
11 GNU Emacs is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
15
16 You should have received a copy of the GNU General Public License
17 along with GNU Emacs; see the file COPYING. If not, write to
18 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
19 Boston, MA 02111-1307, USA. */
20
21
22 #include <signal.h>
23 #include <errno.h>
24
25 #include <config.h>
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
35 #ifdef BSD_SYSTEM
36 #include <sys/ioctl.h>
37 #endif
38
39 #include "lisp.h"
40 #include "commands.h"
41 #include "intervals.h"
42 #include "buffer.h"
43
44 #include "systty.h"
45 #include "blockinput.h"
46 #include "syssignal.h"
47 #include "process.h"
48 #include "keyboard.h"
49
50 #ifdef HAVE_SETRLIMIT
51 #include <sys/time.h>
52 #include <sys/resource.h>
53 #endif
54
55 #ifndef O_RDWR
56 #define O_RDWR 2
57 #endif
58
59 extern void malloc_warning ();
60 extern void set_time_zone_rule ();
61 extern char *index ();
62 extern char *strerror ();
63
64 /* Command line args from shell, as list of strings */
65 Lisp_Object Vcommand_line_args;
66
67 /* The name under which Emacs was invoked, with any leading directory
68 names discarded. */
69 Lisp_Object Vinvocation_name;
70
71 /* The directory name from which Emacs was invoked. */
72 Lisp_Object Vinvocation_directory;
73
74 /* The directory name in which to find subdirs such as lisp and etc.
75 nil means get them only from PATH_LOADSEARCH. */
76 Lisp_Object Vinstallation_directory;
77
78 /* Hook run by `kill-emacs' before it does really anything. */
79 Lisp_Object Vkill_emacs_hook;
80
81 #ifdef SIGUSR1
82 /* Hooks for signal USR1 and USR2 handing */
83 Lisp_Object Vsignal_USR1_hook;
84 #ifdef SIGUSR2
85 Lisp_Object Vsignal_USR2_hook;
86 #endif
87 #endif
88
89 /* Search path separator. */
90 Lisp_Object Vpath_separator;
91
92 /* Set nonzero after Emacs has started up the first time.
93 Prevents reinitialization of the Lisp world and keymaps
94 on subsequent starts. */
95 int initialized;
96
97 #ifdef DOUG_LEA_MALLOC
98 /* Preserves a pointer to the memory allocated that copies that
99 static data inside glibc's malloc. */
100 void *malloc_state_ptr;
101 /* From glibc, a routine that returns a copy of the malloc internal state. */
102 extern void *malloc_get_state ();
103 /* From glibc, a routine that overwrites the malloc internal state. */
104 extern void malloc_set_state ();
105 #endif
106
107 /* Variable whose value is symbol giving operating system type. */
108 Lisp_Object Vsystem_type;
109
110 /* Variable whose value is string giving configuration built for. */
111 Lisp_Object Vsystem_configuration;
112
113 /* Variable whose value is string giving configuration options,
114 for use when reporting bugs. */
115 Lisp_Object Vsystem_configuration_options;
116
117 Lisp_Object Qfile_name_handler_alist;
118
119 /* If non-zero, emacs should not attempt to use an window-specific code,
120 but instead should use the virtual terminal under which it was started */
121 int inhibit_window_system;
122
123 /* If nonzero, set Emacs to run at this priority. This is also used
124 in child_setup and sys_suspend to make sure subshells run at normal
125 priority; Those functions have their own extern declaration. */
126 int emacs_priority;
127
128 /* If non-zero a filter or a sentinel is running. Tested to save the match
129 data on the first attempt to change it inside asynchronous code. */
130 int running_asynch_code;
131
132 #ifdef BSD_PGRPS
133 /* See sysdep.c. */
134 extern int inherited_pgroup;
135 #endif
136
137 #ifdef HAVE_X_WINDOWS
138 /* If non-zero, -d was specified, meaning we're using some window system. */
139 int display_arg;
140 #endif
141
142 /* An address near the bottom of the stack.
143 Tells GC how to save a copy of the stack. */
144 char *stack_bottom;
145
146 #ifdef HAVE_WINDOW_SYSTEM
147 extern Lisp_Object Vwindow_system;
148 #endif /* HAVE_WINDOW_SYSTEM */
149
150 extern Lisp_Object Vauto_save_list_file_name;
151
152 #ifdef USG_SHARED_LIBRARIES
153 /* If nonzero, this is the place to put the end of the writable segment
154 at startup. */
155
156 unsigned int bss_end = 0;
157 #endif
158
159 /* Nonzero means running Emacs without interactive terminal. */
160
161 int noninteractive;
162
163 /* Value of Lisp variable `noninteractive'.
164 Normally same as C variable `noninteractive'
165 but nothing terrible happens if user sets this one. */
166
167 int noninteractive1;
168
169 /* Save argv and argc. */
170 char **initial_argv;
171 int initial_argc;
172
173 static void sort_args ();
174 void syms_of_emacs ();
175 \f
176 /* Signal code for the fatal signal that was received */
177 int fatal_error_code;
178
179 /* Nonzero if handling a fatal error already */
180 int fatal_error_in_progress;
181
182 #ifdef SIGUSR1
183 int SIGUSR1_in_progress=0;
184 SIGTYPE
185 handle_USR1_signal (sig)
186 int sig;
187 {
188 if (! SIGUSR1_in_progress)
189 {
190 SIGUSR1_in_progress = 1;
191
192 if (!NILP (Vrun_hooks) && !noninteractive)
193 call1 (Vrun_hooks, intern ("signal-USR1-hook"));
194
195 SIGUSR1_in_progress = 0;
196 }
197 }
198
199 #ifdef SIGUSR2
200 int SIGUSR2_in_progress=0;
201 SIGTYPE
202 handle_USR2_signal (sig)
203 int sig;
204 {
205 if (! SIGUSR2_in_progress)
206 {
207 SIGUSR2_in_progress = 1;
208
209 if (!NILP (Vrun_hooks) && !noninteractive)
210 call1 (Vrun_hooks, intern ("signal-USR2-hook"));
211
212 SIGUSR2_in_progress = 0;
213 }
214 }
215 #endif
216 #endif
217
218 /* Handle bus errors, illegal instruction, etc. */
219 SIGTYPE
220 fatal_error_signal (sig)
221 int sig;
222 {
223 fatal_error_code = sig;
224 signal (sig, SIG_DFL);
225
226 TOTALLY_UNBLOCK_INPUT;
227
228 /* If fatal error occurs in code below, avoid infinite recursion. */
229 if (! fatal_error_in_progress)
230 {
231 fatal_error_in_progress = 1;
232
233 shut_down_emacs (sig, 0, Qnil);
234 }
235
236 #ifdef VMS
237 LIB$STOP (SS$_ABORT);
238 #else
239 /* Signal the same code; this time it will really be fatal.
240 Remember that since we're in a signal handler, the signal we're
241 going to send is probably blocked, so we have to unblock it if we
242 want to really receive it. */
243 #ifndef MSDOS
244 sigunblock (sigmask (fatal_error_code));
245 #endif
246 kill (getpid (), fatal_error_code);
247 #endif /* not VMS */
248 }
249
250 #ifdef SIGDANGER
251
252 /* Handler for SIGDANGER. */
253 SIGTYPE
254 memory_warning_signal (sig)
255 int sig;
256 {
257 signal (sig, memory_warning_signal);
258
259 malloc_warning ("Operating system warns that virtual memory is running low.\n");
260
261 /* It might be unsafe to call do_auto_save now. */
262 force_auto_save_soon ();
263 }
264 #endif
265
266 /* We define abort, rather than using it from the library,
267 so that GDB can return from a breakpoint here.
268 MSDOS has its own definition on msdos.c */
269
270 #if ! defined (DOS_NT) && ! defined (NO_ABORT)
271 void
272 abort ()
273 {
274 kill (getpid (), SIGABRT);
275 /* This shouldn't be executed, but it prevents a warning. */
276 exit (1);
277 }
278 #endif
279
280 \f
281 /* Code for dealing with Lisp access to the Unix command line */
282
283 static void
284 init_cmdargs (argc, argv, skip_args)
285 int argc;
286 char **argv;
287 int skip_args;
288 {
289 register int i;
290 Lisp_Object name, dir, tem;
291 int count = specpdl_ptr - specpdl;
292 Lisp_Object raw_name;
293
294 initial_argv = argv;
295 initial_argc = argc;
296
297 raw_name = build_string (argv[0]);
298
299 /* Add /: to the front of the name
300 if it would otherwise be treated as magic. */
301 tem = Ffind_file_name_handler (raw_name, Qt);
302 if (! NILP (tem))
303 raw_name = concat2 (build_string ("/:"), raw_name);
304
305 Vinvocation_name = Ffile_name_nondirectory (raw_name);
306 Vinvocation_directory = Ffile_name_directory (raw_name);
307
308 /* If we got no directory in argv[0], search PATH to find where
309 Emacs actually came from. */
310 if (NILP (Vinvocation_directory))
311 {
312 Lisp_Object found;
313 int yes = openp (Vexec_path, Vinvocation_name,
314 EXEC_SUFFIXES, &found, 1);
315 if (yes == 1)
316 {
317 /* Add /: to the front of the name
318 if it would otherwise be treated as magic. */
319 tem = Ffind_file_name_handler (found, Qt);
320 if (! NILP (tem))
321 found = concat2 (build_string ("/:"), found);
322 Vinvocation_directory = Ffile_name_directory (found);
323 }
324 }
325
326 if (!NILP (Vinvocation_directory)
327 && NILP (Ffile_name_absolute_p (Vinvocation_directory)))
328 /* Emacs was started with relative path, like ./emacs.
329 Make it absolute. */
330 Vinvocation_directory = Fexpand_file_name (Vinvocation_directory, Qnil);
331
332 Vinstallation_directory = Qnil;
333
334 if (!NILP (Vinvocation_directory))
335 {
336 dir = Vinvocation_directory;
337 name = Fexpand_file_name (Vinvocation_name, dir);
338 while (1)
339 {
340 Lisp_Object tem, lib_src_exists;
341 Lisp_Object etc_exists, info_exists;
342
343 /* See if dir contains subdirs for use by Emacs.
344 Check for the ones that would exist in a build directory,
345 not including lisp and info. */
346 tem = Fexpand_file_name (build_string ("lib-src"), dir);
347 lib_src_exists = Ffile_exists_p (tem);
348
349 #ifdef MSDOS
350 /* MSDOS installations frequently remove lib-src, but we still
351 must set installation-directory, or else info won't find
352 its files (it uses the value of installation-directory). */
353 tem = Fexpand_file_name (build_string ("info"), dir);
354 info_exists = Ffile_exists_p (tem);
355 #else
356 info_exists = Qnil;
357 #endif
358
359 if (!NILP (lib_src_exists) || !NILP (info_exists))
360 {
361 tem = Fexpand_file_name (build_string ("etc"), dir);
362 etc_exists = Ffile_exists_p (tem);
363 if (!NILP (etc_exists))
364 {
365 Vinstallation_directory
366 = Ffile_name_as_directory (dir);
367 break;
368 }
369 }
370
371 /* See if dir's parent contains those subdirs. */
372 tem = Fexpand_file_name (build_string ("../lib-src"), dir);
373 lib_src_exists = Ffile_exists_p (tem);
374
375
376 #ifdef MSDOS
377 /* See the MSDOS commentary above. */
378 tem = Fexpand_file_name (build_string ("../info"), dir);
379 info_exists = Ffile_exists_p (tem);
380 #else
381 info_exists = Qnil;
382 #endif
383
384 if (!NILP (lib_src_exists) || !NILP (info_exists))
385 {
386 tem = Fexpand_file_name (build_string ("../etc"), dir);
387 etc_exists = Ffile_exists_p (tem);
388 if (!NILP (etc_exists))
389 {
390 tem = Fexpand_file_name (build_string (".."), dir);
391 Vinstallation_directory
392 = Ffile_name_as_directory (tem);
393 break;
394 }
395 }
396
397 /* If the Emacs executable is actually a link,
398 next try the dir that the link points into. */
399 tem = Ffile_symlink_p (name);
400 if (!NILP (tem))
401 {
402 name = Fexpand_file_name (tem, dir);
403 dir = Ffile_name_directory (name);
404 }
405 else
406 break;
407 }
408 }
409
410 Vcommand_line_args = Qnil;
411
412 for (i = argc - 1; i >= 0; i--)
413 {
414 if (i == 0 || i > skip_args)
415 Vcommand_line_args
416 = Fcons (build_string (argv[i]), Vcommand_line_args);
417 }
418
419 unbind_to (count, Qnil);
420 }
421
422 DEFUN ("invocation-name", Finvocation_name, Sinvocation_name, 0, 0, 0,
423 "Return the program name that was used to run Emacs.\n\
424 Any directory names are omitted.")
425 ()
426 {
427 return Fcopy_sequence (Vinvocation_name);
428 }
429
430 DEFUN ("invocation-directory", Finvocation_directory, Sinvocation_directory,
431 0, 0, 0,
432 "Return the directory name in which the Emacs executable was located")
433 ()
434 {
435 return Fcopy_sequence (Vinvocation_directory);
436 }
437
438 \f
439 #ifdef VMS
440 #ifdef LINK_CRTL_SHARE
441 #ifdef SHARABLE_LIB_BUG
442 extern noshare char **environ;
443 #endif /* SHARABLE_LIB_BUG */
444 #endif /* LINK_CRTL_SHARE */
445 #endif /* VMS */
446
447 #ifdef HAVE_TZSET
448 /* A valid but unlikely value for the TZ environment value.
449 It is OK (though a bit slower) if the user actually chooses this value. */
450 static char dump_tz[] = "UtC0";
451 #endif
452
453 #ifndef ORDINARY_LINK
454 /* We don't include crtbegin.o and crtend.o in the link,
455 so these functions and variables might be missed.
456 Provide dummy definitions to avoid error.
457 (We don't have any real constructors or destructors.) */
458 #ifdef __GNUC__
459 #ifndef GCC_CTORS_IN_LIBC
460 void __do_global_ctors ()
461 {}
462 void __do_global_ctors_aux ()
463 {}
464 void __do_global_dtors ()
465 {}
466 /* Linux has a bug in its library; avoid an error. */
467 #ifndef LINUX
468 char * __CTOR_LIST__[2] = { (char *) (-1), 0 };
469 #endif
470 char * __DTOR_LIST__[2] = { (char *) (-1), 0 };
471 #endif /* GCC_CTORS_IN_LIBC */
472 void __main ()
473 {}
474 #endif /* __GNUC__ */
475 #endif /* ORDINARY_LINK */
476
477 /* Test whether the next argument in ARGV matches SSTR or a prefix of
478 LSTR (at least MINLEN characters). If so, then if VALPTR is non-null
479 (the argument is supposed to have a value) store in *VALPTR either
480 the next argument or the portion of this one after the equal sign.
481 ARGV is read starting at position *SKIPPTR; this index is advanced
482 by the number of arguments used.
483
484 Too bad we can't just use getopt for all of this, but we don't have
485 enough information to do it right. */
486
487 static int
488 argmatch (argv, argc, sstr, lstr, minlen, valptr, skipptr)
489 char **argv;
490 int argc;
491 char *sstr;
492 char *lstr;
493 int minlen;
494 char **valptr;
495 int *skipptr;
496 {
497 char *p;
498 int arglen;
499 char *arg;
500
501 /* Don't access argv[argc]; give up in advance. */
502 if (argc <= *skipptr + 1)
503 return 0;
504
505 arg = argv[*skipptr+1];
506 if (arg == NULL)
507 return 0;
508 if (strcmp (arg, sstr) == 0)
509 {
510 if (valptr != NULL)
511 {
512 *valptr = argv[*skipptr+2];
513 *skipptr += 2;
514 }
515 else
516 *skipptr += 1;
517 return 1;
518 }
519 arglen = (valptr != NULL && (p = index (arg, '=')) != NULL
520 ? p - arg : strlen (arg));
521 if (lstr == 0 || arglen < minlen || strncmp (arg, lstr, arglen) != 0)
522 return 0;
523 else if (valptr == NULL)
524 {
525 *skipptr += 1;
526 return 1;
527 }
528 else if (p != NULL)
529 {
530 *valptr = p+1;
531 *skipptr += 1;
532 return 1;
533 }
534 else if (argv[*skipptr+2] != NULL)
535 {
536 *valptr = argv[*skipptr+2];
537 *skipptr += 2;
538 return 1;
539 }
540 else
541 {
542 return 0;
543 }
544 }
545
546 /* ARGSUSED */
547 int
548 main (argc, argv, envp)
549 int argc;
550 char **argv;
551 char **envp;
552 {
553 char stack_bottom_variable;
554 int skip_args = 0;
555 extern int errno;
556 extern int sys_nerr;
557 #ifdef HAVE_SETRLIMIT
558 struct rlimit rlim;
559 #endif
560
561 #ifdef LINUX_SBRK_BUG
562 __sbrk (1);
563 #endif
564
565 #ifdef DOUG_LEA_MALLOC
566 if (initialized)
567 {
568 extern void r_alloc_reinit ();
569 malloc_set_state (malloc_state_ptr);
570 free (malloc_state_ptr);
571 r_alloc_reinit ();
572 }
573 #endif
574
575 #ifdef RUN_TIME_REMAP
576 if (initialized)
577 run_time_remap (argv[0]);
578 #endif
579
580 sort_args (argc, argv);
581
582 if (argmatch (argv, argc, "-version", "--version", 3, NULL, &skip_args))
583 {
584 Lisp_Object tem;
585 tem = Fsymbol_value (intern ("emacs-version"));
586 if (!STRINGP (tem))
587 {
588 fprintf (stderr, "Invalid value of `emacs-version'\n");
589 exit (1);
590 }
591 else
592 {
593 printf ("GNU Emacs %s\n", XSTRING (tem)->data);
594 printf ("Copyright (C) 1997 Free Software Foundation, Inc.\n");
595 printf ("GNU Emacs comes with ABSOLUTELY NO WARRANTY.\n");
596 printf ("You may redistribute copies of Emacs\n");
597 printf ("under the terms of the GNU General Public License.\n");
598 printf ("For more information about these matters, ");
599 printf ("see the files named COPYING.\n");
600 exit (0);
601 }
602 }
603
604 /* Map in shared memory, if we are using that. */
605 #ifdef HAVE_SHM
606 if (argmatch (argv, argc, "-nl", "--no-shared-memory", 6, NULL, &skip_args))
607 {
608 map_in_data (0);
609 /* The shared memory was just restored, which clobbered this. */
610 skip_args = 1;
611 }
612 else
613 {
614 map_in_data (1);
615 /* The shared memory was just restored, which clobbered this. */
616 skip_args = 0;
617 }
618 #endif
619
620 #ifdef NeXT
621 {
622 extern int malloc_cookie;
623 /* This helps out unexnext.c. */
624 if (initialized)
625 if (malloc_jumpstart (malloc_cookie) != 0)
626 printf ("malloc jumpstart failed!\n");
627 }
628 #endif /* NeXT */
629
630 #ifdef VMS
631 /* If -map specified, map the data file in */
632 {
633 char *file;
634 if (argmatch (argv, argc, "-map", "--map-data", 3, &mapin_file, &skip_args))
635 mapin_data (file);
636 }
637
638 #ifdef LINK_CRTL_SHARE
639 #ifdef SHARABLE_LIB_BUG
640 /* Bletcherous shared libraries! */
641 if (!stdin)
642 stdin = fdopen (0, "r");
643 if (!stdout)
644 stdout = fdopen (1, "w");
645 if (!stderr)
646 stderr = fdopen (2, "w");
647 if (!environ)
648 environ = envp;
649 #endif /* SHARABLE_LIB_BUG */
650 #endif /* LINK_CRTL_SHARE */
651 #endif /* VMS */
652
653 #if defined (HAVE_SETRLIMIT) && defined (RLIMIT_STACK)
654 /* Extend the stack space available.
655 Don't do that if dumping, since some systems (e.g. DJGPP)
656 might define a smaller stack limit at that time. */
657 if (1
658 #ifndef CANNOT_DUMP
659 && (!noninteractive || initialized)
660 #endif
661 && !getrlimit (RLIMIT_STACK, &rlim))
662 {
663 long newlim;
664 extern int re_max_failures;
665 /* Approximate the amount regex.c needs per unit of re_max_failures. */
666 int ratio = 20 * sizeof (char *);
667 /* Then add 33% to cover the size of the smaller stacks that regex.c
668 successively allocates and discards, on its way to the maximum. */
669 ratio += ratio / 3;
670 /* Add in some extra to cover
671 what we're likely to use for other reasons. */
672 newlim = re_max_failures * ratio + 200000;
673 #ifdef __NetBSD__
674 /* NetBSD (at least NetBSD 1.2G and former) has a bug in its
675 stack allocation routine for new process that the allocation
676 fails if stack limit is not on page boundary. So, round up the
677 new limit to page boundary. */
678 newlim = (newlim + getpagesize () - 1) / getpagesize () * getpagesize();
679 #endif
680 if (newlim > rlim.rlim_max)
681 {
682 newlim = rlim.rlim_max;
683 /* Don't let regex.c overflow the stack we have. */
684 re_max_failures = (newlim - 200000) / ratio;
685 }
686 if (rlim.rlim_cur < newlim)
687 rlim.rlim_cur = newlim;
688
689 setrlimit (RLIMIT_STACK, &rlim);
690 }
691 #endif /* HAVE_SETRLIMIT and RLIMIT_STACK */
692
693 /* Record (approximately) where the stack begins. */
694 stack_bottom = &stack_bottom_variable;
695
696 #ifdef USG_SHARED_LIBRARIES
697 if (bss_end)
698 brk ((void *)bss_end);
699 #endif
700
701 clearerr (stdin);
702
703 #ifndef SYSTEM_MALLOC
704 if (! initialized)
705 {
706 /* Arrange to get warning messages as memory fills up. */
707 memory_warnings (0, malloc_warning);
708
709 /* Arrange to disable interrupt input while malloc and friends are
710 running. */
711 uninterrupt_malloc ();
712 }
713 #endif /* not SYSTEM_MALLOC */
714
715 #ifdef MSDOS
716 /* We do all file input/output as binary files. When we need to translate
717 newlines, we do that manually. */
718 _fmode = O_BINARY;
719
720 #if __DJGPP__ >= 2
721 if (!isatty (fileno (stdin)))
722 setmode (fileno (stdin), O_BINARY);
723 if (!isatty (fileno (stdout)))
724 {
725 fflush (stdout);
726 setmode (fileno (stdout), O_BINARY);
727 }
728 #else /* not __DJGPP__ >= 2 */
729 (stdin)->_flag &= ~_IOTEXT;
730 (stdout)->_flag &= ~_IOTEXT;
731 (stderr)->_flag &= ~_IOTEXT;
732 #endif /* not __DJGPP__ >= 2 */
733 #endif /* MSDOS */
734
735 #ifdef SET_EMACS_PRIORITY
736 if (emacs_priority)
737 nice (emacs_priority);
738 setuid (getuid ());
739 #endif /* SET_EMACS_PRIORITY */
740
741 #ifdef EXTRA_INITIALIZE
742 EXTRA_INITIALIZE;
743 #endif
744
745 inhibit_window_system = 0;
746
747 /* Handle the -t switch, which specifies filename to use as terminal */
748 {
749 char *term;
750 if (argmatch (argv, argc, "-t", "--terminal", 4, &term, &skip_args))
751 {
752 int result;
753 close (0);
754 close (1);
755 result = open (term, O_RDWR, 2 );
756 if (result < 0)
757 {
758 char *errstring = strerror (errno);
759 fprintf (stderr, "emacs: %s: %s\n", term, errstring);
760 exit (1);
761 }
762 dup (0);
763 if (! isatty (0))
764 {
765 fprintf (stderr, "emacs: %s: not a tty\n", term);
766 exit (1);
767 }
768 fprintf (stderr, "Using %s\n", term);
769 #ifdef HAVE_WINDOW_SYSTEM
770 inhibit_window_system = 1; /* -t => -nw */
771 #endif
772 }
773 }
774 if (argmatch (argv, argc, "-nw", "--no-windows", 6, NULL, &skip_args))
775 inhibit_window_system = 1;
776
777 /* Handle the -batch switch, which means don't do interactive display. */
778 noninteractive = 0;
779 if (argmatch (argv, argc, "-batch", "--batch", 5, NULL, &skip_args))
780 noninteractive = 1;
781
782 /* Handle the --help option, which gives a usage message.. */
783 if (argmatch (argv, argc, "-help", "--help", 3, NULL, &skip_args))
784 {
785 printf ("\
786 Usage: %s [-t term] [--terminal term] [-nw] [--no-windows] [--batch]\n\
787 [-q] [--no-init-file] [-u user] [--user user] [--debug-init]\n\
788 [--unibyte] [--multibyte] [--version] [--no-site-file]\n\
789 [-f func] [--funcall func] [-l file] [--load file] [--insert file]\n\
790 [+linenum] file-to-visit [--kill]\n\
791 Report bugs to bug-gnu-emacs@gnu.org. First, please see\n\
792 the Bugs section of the Emacs manual or the file BUGS.\n", argv[0]);
793 exit (0);
794 }
795
796 #ifdef HAVE_X_WINDOWS
797 /* Stupid kludge to catch command-line display spec. We can't
798 handle this argument entirely in window system dependent code
799 because we don't even know which window system dependent code
800 to run until we've recognized this argument. */
801 {
802 char *displayname = 0;
803 int i;
804 int count_before = skip_args;
805
806 if (argmatch (argv, argc, "-d", "--display", 3, &displayname, &skip_args))
807 display_arg = 1;
808 else if (argmatch (argv, argc, "-display", 0, 3, &displayname, &skip_args))
809 display_arg = 1;
810
811 /* If we have the form --display=NAME,
812 convert it into -d name.
813 This requires inserting a new element into argv. */
814 if (displayname != 0 && skip_args - count_before == 1)
815 {
816 char **new = (char **) xmalloc (sizeof (char *) * (argc + 2));
817 int j;
818
819 for (j = 0; j < count_before + 1; j++)
820 new[j] = argv[j];
821 new[count_before + 1] = "-d";
822 new[count_before + 2] = displayname;
823 for (j = count_before + 2; j <argc; j++)
824 new[j + 1] = argv[j];
825 argv = new;
826 argc++;
827 }
828 /* Change --display to -d, when its arg is separate. */
829 else if (displayname != 0 && skip_args > count_before
830 && argv[count_before + 1][1] == '-')
831 argv[count_before + 1] = "-d";
832
833 /* Don't actually discard this arg. */
834 skip_args = count_before;
835 }
836 #endif
837
838 if (! noninteractive)
839 {
840 #ifdef BSD_PGRPS
841 if (initialized)
842 {
843 inherited_pgroup = EMACS_GETPGRP (0);
844 setpgrp (0, getpid ());
845 }
846 #else
847 #if defined (USG5) && defined (INTERRUPT_INPUT)
848 setpgrp ();
849 #endif
850 #endif
851 }
852
853 #ifdef POSIX_SIGNALS
854 init_signals ();
855 #endif
856
857 /* Don't catch SIGHUP if dumping. */
858 if (1
859 #ifndef CANNOT_DUMP
860 && initialized
861 #endif
862 )
863 {
864 sigblock (sigmask (SIGHUP));
865 /* In --batch mode, don't catch SIGHUP if already ignored.
866 That makes nohup work. */
867 if (! noninteractive
868 || signal (SIGHUP, SIG_IGN) != SIG_IGN)
869 signal (SIGHUP, fatal_error_signal);
870 sigunblock (sigmask (SIGHUP));
871 }
872
873 if (
874 #ifndef CANNOT_DUMP
875 ! noninteractive || initialized
876 #else
877 1
878 #endif
879 )
880 {
881 /* Don't catch these signals in batch mode if dumping.
882 On some machines, this sets static data that would make
883 signal fail to work right when the dumped Emacs is run. */
884 signal (SIGQUIT, fatal_error_signal);
885 signal (SIGILL, fatal_error_signal);
886 signal (SIGTRAP, fatal_error_signal);
887 #ifdef SIGUSR1
888 signal (SIGUSR1, handle_USR1_signal);
889 #ifdef SIGUSR2
890 signal (SIGUSR2, handle_USR2_signal);
891 #endif
892 #endif
893 #ifdef SIGABRT
894 signal (SIGABRT, fatal_error_signal);
895 #endif
896 #ifdef SIGHWE
897 signal (SIGHWE, fatal_error_signal);
898 #endif
899 #ifdef SIGPRE
900 signal (SIGPRE, fatal_error_signal);
901 #endif
902 #ifdef SIGORE
903 signal (SIGORE, fatal_error_signal);
904 #endif
905 #ifdef SIGUME
906 signal (SIGUME, fatal_error_signal);
907 #endif
908 #ifdef SIGDLK
909 signal (SIGDLK, fatal_error_signal);
910 #endif
911 #ifdef SIGCPULIM
912 signal (SIGCPULIM, fatal_error_signal);
913 #endif
914 #ifdef SIGIOT
915 /* This is missing on some systems - OS/2, for example. */
916 signal (SIGIOT, fatal_error_signal);
917 #endif
918 #ifdef SIGEMT
919 signal (SIGEMT, fatal_error_signal);
920 #endif
921 signal (SIGFPE, fatal_error_signal);
922 #ifdef SIGBUS
923 signal (SIGBUS, fatal_error_signal);
924 #endif
925 signal (SIGSEGV, fatal_error_signal);
926 #ifdef SIGSYS
927 signal (SIGSYS, fatal_error_signal);
928 #endif
929 signal (SIGTERM, fatal_error_signal);
930 #ifdef SIGXCPU
931 signal (SIGXCPU, fatal_error_signal);
932 #endif
933 #ifdef SIGXFSZ
934 signal (SIGXFSZ, fatal_error_signal);
935 #endif /* SIGXFSZ */
936
937 #ifdef SIGDANGER
938 /* This just means available memory is getting low. */
939 signal (SIGDANGER, memory_warning_signal);
940 #endif
941
942 #ifdef AIX
943 /* 20 is SIGCHLD, 21 is SIGTTIN, 22 is SIGTTOU. */
944 signal (SIGXCPU, fatal_error_signal);
945 #ifndef _I386
946 signal (SIGIOINT, fatal_error_signal);
947 #endif
948 signal (SIGGRANT, fatal_error_signal);
949 signal (SIGRETRACT, fatal_error_signal);
950 signal (SIGSOUND, fatal_error_signal);
951 signal (SIGMSG, fatal_error_signal);
952 #endif /* AIX */
953 }
954
955 noninteractive1 = noninteractive;
956
957 /* Perform basic initializations (not merely interning symbols) */
958
959 if (!initialized)
960 {
961 init_alloc_once ();
962 init_obarray ();
963 init_eval_once ();
964 init_charset_once ();
965 init_coding_once ();
966 init_syntax_once (); /* Create standard syntax table. */
967 init_category_once (); /* Create standard category table. */
968 /* Must be done before init_buffer */
969 init_casetab_once ();
970 init_buffer_once (); /* Create buffer table and some buffers */
971 init_minibuf_once (); /* Create list of minibuffers */
972 /* Must precede init_window_once */
973 init_window_once (); /* Init the window system */
974 }
975
976 init_alloc ();
977 init_eval ();
978 init_data ();
979 running_asynch_code = 0;
980
981 /* Handle --unibyte and the EMACS_UNIBYTE envvar,
982 but not while dumping. */
983 if (
984 #ifndef CANNOT_DUMP
985 ! noninteractive || initialized
986 #else
987 1
988 #endif
989 )
990 {
991 int inhibit_unibyte = 0;
992
993 /* --multibyte overrides EMACS_UNIBYTE. */
994 if (argmatch (argv, argc, "-no-unibyte", "--no-unibyte", 4, NULL, &skip_args)
995 || argmatch (argv, argc, "-multibyte", "--multibyte", 4, NULL, &skip_args))
996 inhibit_unibyte = 1;
997
998 /* --unibyte requests that we set up to do everything with single-byte
999 buffers and strings. We need to handle this before calling
1000 init_lread, init_editfns and other places that generate Lisp strings
1001 from text in the environment. */
1002 if (argmatch (argv, argc, "-unibyte", "--unibyte", 4, NULL, &skip_args)
1003 || argmatch (argv, argc, "-no-multibyte", "--no-multibyte", 4, NULL, &skip_args)
1004 || (getenv ("EMACS_UNIBYTE") && !inhibit_unibyte))
1005 {
1006 Lisp_Object old_log_max;
1007 Lisp_Object symbol, tail;
1008
1009 symbol = intern ("default-enable-multibyte-characters");
1010 Fset (symbol, Qnil);
1011
1012 /* Erase pre-dump messages in *Messages* now so no abort. */
1013 old_log_max = Vmessage_log_max;
1014 XSETFASTINT (Vmessage_log_max, 0);
1015 message_dolog ("", 0, 1, 0);
1016 Vmessage_log_max = old_log_max;
1017
1018 for (tail = Vbuffer_alist; CONSP (tail);
1019 tail = XCONS (tail)->cdr)
1020 {
1021 Lisp_Object buffer;
1022
1023 buffer = Fcdr (XCONS (tail)->car);
1024 /* Verify that all buffers are empty now, as they
1025 ought to be. */
1026 if (BUF_Z (XBUFFER (buffer)) > BUF_BEG (XBUFFER (buffer)))
1027 abort ();
1028 /* It is safe to do this crudely in an empty buffer. */
1029 XBUFFER (buffer)->enable_multibyte_characters = Qnil;
1030 }
1031 }
1032 }
1033
1034 #ifdef MSDOS
1035 /* Call early 'cause init_environment needs it. */
1036 init_dosfns ();
1037 /* Set defaults for several environment variables. */
1038 if (initialized)
1039 init_environment (argc, argv, skip_args);
1040 else
1041 tzset ();
1042 #endif /* MSDOS */
1043
1044 #ifdef WINDOWSNT
1045 /* Initialize environment from registry settings. */
1046 init_environment ();
1047 init_ntproc (); /* must precede init_editfns */
1048 #endif
1049
1050 /* egetenv is a pretty low-level facility, which may get called in
1051 many circumstances; it seems flimsy to put off initializing it
1052 until calling init_callproc. */
1053 set_process_environment ();
1054 /* AIX crashes are reported in system versions 3.2.3 and 3.2.4
1055 if this is not done. Do it after set_process_environment so that we
1056 don't pollute Vprocess_environment. */
1057 #ifdef AIX
1058 putenv ("LANG=C");
1059 #endif
1060
1061 init_buffer (); /* Init default directory of main buffer */
1062
1063 init_callproc_1 (); /* Must precede init_cmdargs and init_sys_modes. */
1064 init_cmdargs (argc, argv, skip_args); /* Must precede init_lread. */
1065
1066 if (initialized)
1067 {
1068 /* Erase any pre-dump messages in the message log, to avoid confusion */
1069 Lisp_Object old_log_max;
1070 old_log_max = Vmessage_log_max;
1071 XSETFASTINT (Vmessage_log_max, 0);
1072 message_dolog ("", 0, 1, 0);
1073 Vmessage_log_max = old_log_max;
1074 }
1075
1076 init_callproc (); /* Must follow init_cmdargs but not init_sys_modes. */
1077 init_lread ();
1078
1079 if (!noninteractive)
1080 {
1081 #ifdef VMS
1082 init_vms_input ();/* init_display calls get_frame_size, that needs this */
1083 #endif /* VMS */
1084 init_display (); /* Determine terminal type. init_sys_modes uses results */
1085 }
1086 init_keyboard (); /* This too must precede init_sys_modes */
1087 #ifdef VMS
1088 init_vmsproc (); /* And this too. */
1089 #endif /* VMS */
1090 init_sys_modes (); /* Init system terminal modes (RAW or CBREAK, etc.) */
1091 init_xdisp ();
1092 init_macros ();
1093 init_editfns ();
1094 #ifdef LISP_FLOAT_TYPE
1095 init_floatfns ();
1096 #endif
1097 #ifdef VMS
1098 init_vmsfns ();
1099 #endif /* VMS */
1100 init_process ();
1101
1102 /* Intern the names of all standard functions and variables; define standard keys */
1103
1104 if (!initialized)
1105 {
1106 /* The basic levels of Lisp must come first */
1107 /* And data must come first of all
1108 for the sake of symbols like error-message */
1109 syms_of_data ();
1110 syms_of_alloc ();
1111 syms_of_lread ();
1112 syms_of_print ();
1113 syms_of_eval ();
1114 syms_of_fns ();
1115 syms_of_floatfns ();
1116
1117 syms_of_abbrev ();
1118 syms_of_buffer ();
1119 syms_of_bytecode ();
1120 syms_of_callint ();
1121 syms_of_casefiddle ();
1122 syms_of_casetab ();
1123 syms_of_callproc ();
1124 syms_of_category ();
1125 syms_of_ccl ();
1126 syms_of_charset ();
1127 syms_of_cmds ();
1128 #ifndef NO_DIR_LIBRARY
1129 syms_of_dired ();
1130 #endif /* not NO_DIR_LIBRARY */
1131 syms_of_display ();
1132 syms_of_doc ();
1133 syms_of_editfns ();
1134 syms_of_emacs ();
1135 syms_of_fileio ();
1136 syms_of_coding (); /* This should be after syms_of_fileio. */
1137 #ifdef CLASH_DETECTION
1138 syms_of_filelock ();
1139 #endif /* CLASH_DETECTION */
1140 syms_of_indent ();
1141 syms_of_insdel ();
1142 syms_of_keyboard ();
1143 syms_of_keymap ();
1144 syms_of_macros ();
1145 syms_of_marker ();
1146 syms_of_minibuf ();
1147 syms_of_mocklisp ();
1148 syms_of_process ();
1149 syms_of_search ();
1150 syms_of_frame ();
1151 syms_of_syntax ();
1152 syms_of_term ();
1153 syms_of_undo ();
1154
1155 /* Only defined if Emacs is compiled with USE_TEXT_PROPERTIES */
1156 syms_of_textprop ();
1157 #ifdef VMS
1158 syms_of_vmsproc ();
1159 #endif /* VMS */
1160 #ifdef WINDOWSNT
1161 syms_of_ntproc ();
1162 #endif /* WINDOWSNT */
1163 syms_of_window ();
1164 syms_of_xdisp ();
1165 #ifdef HAVE_X_WINDOWS
1166 syms_of_xterm ();
1167 syms_of_xfns ();
1168 syms_of_fontset ();
1169 #ifdef HAVE_X11
1170 syms_of_xselect ();
1171 #endif
1172 #endif /* HAVE_X_WINDOWS */
1173
1174 #ifndef HAVE_NTGUI
1175 syms_of_xfaces ();
1176 syms_of_xmenu ();
1177 #endif
1178
1179 #ifdef HAVE_NTGUI
1180 syms_of_w32term ();
1181 syms_of_w32fns ();
1182 syms_of_w32faces ();
1183 syms_of_w32select ();
1184 syms_of_w32menu ();
1185 #endif /* HAVE_NTGUI */
1186
1187 #ifdef SYMS_SYSTEM
1188 SYMS_SYSTEM;
1189 #endif
1190
1191 #ifdef SYMS_MACHINE
1192 SYMS_MACHINE;
1193 #endif
1194
1195 keys_of_casefiddle ();
1196 keys_of_cmds ();
1197 keys_of_buffer ();
1198 keys_of_keyboard ();
1199 keys_of_keymap ();
1200 keys_of_macros ();
1201 keys_of_minibuf ();
1202 keys_of_window ();
1203 keys_of_frame ();
1204 }
1205
1206 if (!initialized)
1207 {
1208 char *file;
1209 /* Handle -l loadup-and-dump, args passed by Makefile. */
1210 if (argmatch (argv, argc, "-l", "--load", 3, &file, &skip_args))
1211 Vtop_level = Fcons (intern ("load"),
1212 Fcons (build_string (file), Qnil));
1213 #ifdef CANNOT_DUMP
1214 /* Unless next switch is -nl, load "loadup.el" first thing. */
1215 if (!argmatch (argv, argc, "-nl", "--no-loadup", 6, NULL, &skip_args))
1216 Vtop_level = Fcons (intern ("load"),
1217 Fcons (build_string ("loadup.el"), Qnil));
1218 #endif /* CANNOT_DUMP */
1219 }
1220
1221 if (initialized)
1222 {
1223 #ifdef HAVE_TZSET
1224 {
1225 /* If the execution TZ happens to be the same as the dump TZ,
1226 change it to some other value and then change it back,
1227 to force the underlying implementation to reload the TZ info.
1228 This is needed on implementations that load TZ info from files,
1229 since the TZ file contents may differ between dump and execution. */
1230 char *tz = getenv ("TZ");
1231 if (tz && !strcmp (tz, dump_tz))
1232 {
1233 ++*tz;
1234 tzset ();
1235 --*tz;
1236 }
1237 }
1238 #endif
1239 }
1240
1241 /* Gerd Moellmann <gerd@acm.org> says this makes profiling work on
1242 FreeBSD. It might work on some other systems too.
1243 Give it a try and tell me if it works on your system. */
1244 #ifdef __FreeBSD__
1245 #ifdef PROFILING
1246 if (initialized)
1247 {
1248 extern void _mcleanup ();
1249 extern char etext;
1250 extern Lisp_Object Fredraw_frame ();
1251 atexit (_mcleanup);
1252 /* This uses Fredraw_frame because that function
1253 comes first in the Emacs executable.
1254 It might be better to use something that gives
1255 the start of the text segment, but start_of_text
1256 is not defined on all systems now. */
1257 monstartup (Fredraw_frame, &etext);
1258 }
1259 else
1260 moncontrol (0);
1261 #endif
1262 #endif
1263
1264 initialized = 1;
1265
1266 #ifdef LOCALTIME_CACHE
1267 /* Some versions of localtime have a bug. They cache the value of the time
1268 zone rather than looking it up every time. Since localtime() is
1269 called to bolt the undumping time into the undumped emacs, this
1270 results in localtime ignoring the TZ environment variable.
1271 This flushes the new TZ value into localtime. */
1272 tzset ();
1273 #endif /* defined (LOCALTIME_CACHE) */
1274
1275 /* Enter editor command loop. This never returns. */
1276 Frecursive_edit ();
1277 /* NOTREACHED */
1278 }
1279 \f
1280 /* Sort the args so we can find the most important ones
1281 at the beginning of argv. */
1282
1283 /* First, here's a table of all the standard options. */
1284
1285 struct standard_args
1286 {
1287 char *name;
1288 char *longname;
1289 int priority;
1290 int nargs;
1291 };
1292
1293 struct standard_args standard_args[] =
1294 {
1295 { "-version", "--version", 110, 0 },
1296 { "-help", "--help", 110, 0 },
1297 { "-nl", "--no-shared-memory", 100, 0 },
1298 #ifdef VMS
1299 { "-map", "--map-data", 100, 0 },
1300 #endif
1301 { "-t", "--terminal", 90, 1 },
1302 { "-d", "--display", 80, 1 },
1303 { "-display", 0, 80, 1 },
1304 { "-nw", "--no-windows", 70, 0 },
1305 { "-no-unibyte", "--no-unibyte", 66, 0 },
1306 { "-multibyte", "--multibyte", 66, 0 },
1307 { "-unibyte", "--unibyte", 65, 0 },
1308 { "-no-multibyte", "--no-multibyte", 65, 0 },
1309 { "-batch", "--batch", 60, 0 },
1310 { "-q", "--no-init-file", 50, 0 },
1311 { "-no-init-file", 0, 50, 0 },
1312 { "-no-site-file", "--no-site-file", 40, 0 },
1313 { "-u", "--user", 30, 1 },
1314 { "-user", 0, 30, 1 },
1315 { "-debug-init", "--debug-init", 20, 0 },
1316 { "-i", "--icon-type", 15, 0 },
1317 { "-itype", 0, 15, 0 },
1318 { "-iconic", "--iconic", 15, 0 },
1319 { "-bg", "--background-color", 10, 1 },
1320 { "-background", 0, 10, 1 },
1321 { "-fg", "--foreground-color", 10, 1 },
1322 { "-foreground", 0, 10, 1 },
1323 { "-bd", "--border-color", 10, 1 },
1324 { "-bw", "--border-width", 10, 1 },
1325 { "-ib", "--internal-border", 10, 1 },
1326 { "-ms", "--mouse-color", 10, 1 },
1327 { "-cr", "--cursor-color", 10, 1 },
1328 { "-fn", "--font", 10, 1 },
1329 { "-font", 0, 10, 1 },
1330 { "-g", "--geometry", 10, 1 },
1331 { "-geometry", 0, 10, 1 },
1332 { "-T", "--title", 10, 1 },
1333 { "-title", 0, 10, 1 },
1334 { "-name", "--name", 10, 1 },
1335 { "-xrm", "--xrm", 10, 1 },
1336 { "-r", "--reverse-video", 5, 0 },
1337 { "-rv", 0, 5, 0 },
1338 { "-reverse", 0, 5, 0 },
1339 { "-hb", "--horizontal-scroll-bars", 5, 0 },
1340 { "-vb", "--vertical-scroll-bars", 5, 0 },
1341 /* These have the same priority as ordinary file name args,
1342 so they are not reordered with respect to those. */
1343 { "-L", "--directory", 0, 1 },
1344 { "-directory", 0, 0, 1 },
1345 { "-l", "--load", 0, 1 },
1346 { "-load", 0, 0, 1 },
1347 { "-f", "--funcall", 0, 1 },
1348 { "-funcall", 0, 0, 1 },
1349 { "-eval", "--eval", 0, 1 },
1350 { "-find-file", "--find-file", 0, 1 },
1351 { "-visit", "--visit", 0, 1 },
1352 { "-insert", "--insert", 0, 1 },
1353 /* This should be processed after ordinary file name args and the like. */
1354 { "-kill", "--kill", -10, 0 },
1355 };
1356
1357 /* Reorder the elements of ARGV (assumed to have ARGC elements)
1358 so that the highest priority ones come first.
1359 Do not change the order of elements of equal priority.
1360 If an option takes an argument, keep it and its argument together. */
1361
1362 static void
1363 sort_args (argc, argv)
1364 int argc;
1365 char **argv;
1366 {
1367 char **new = (char **) xmalloc (sizeof (char *) * argc);
1368 /* For each element of argv,
1369 the corresponding element of options is:
1370 0 for an option that takes no arguments,
1371 1 for an option that takes one argument, etc.
1372 -1 for an ordinary non-option argument. */
1373 int *options = (int *) xmalloc (sizeof (int) * argc);
1374 int *priority = (int *) xmalloc (sizeof (int) * argc);
1375 int to = 1;
1376 int from;
1377 int i;
1378 int end_of_options = argc;
1379
1380 /* Categorize all the options,
1381 and figure out which argv elts are option arguments. */
1382 for (from = 1; from < argc; from++)
1383 {
1384 options[from] = -1;
1385 priority[from] = 0;
1386 if (argv[from][0] == '-')
1387 {
1388 int match, thislen;
1389 char *equals;
1390
1391 /* If we have found "--", don't consider
1392 any more arguments as options. */
1393 if (argv[from][1] == '-' && argv[from][2] == 0)
1394 {
1395 /* Leave the "--", and everything following it, at the end. */
1396 for (; from < argc; from++)
1397 {
1398 priority[from] = -100;
1399 options[from] = -1;
1400 }
1401 break;
1402 }
1403
1404 /* Look for a match with a known old-fashioned option. */
1405 for (i = 0; i < sizeof (standard_args) / sizeof (standard_args[0]); i++)
1406 if (!strcmp (argv[from], standard_args[i].name))
1407 {
1408 options[from] = standard_args[i].nargs;
1409 priority[from] = standard_args[i].priority;
1410 if (from + standard_args[i].nargs >= argc)
1411 fatal ("Option `%s' requires an argument\n", argv[from]);
1412 from += standard_args[i].nargs;
1413 goto done;
1414 }
1415
1416 /* Look for a match with a known long option.
1417 MATCH is -1 if no match so far, -2 if two or more matches so far,
1418 >= 0 (the table index of the match) if just one match so far. */
1419 if (argv[from][1] == '-')
1420 {
1421 match = -1;
1422 thislen = strlen (argv[from]);
1423 equals = index (argv[from], '=');
1424 if (equals != 0)
1425 thislen = equals - argv[from];
1426
1427 for (i = 0;
1428 i < sizeof (standard_args) / sizeof (standard_args[0]); i++)
1429 if (standard_args[i].longname
1430 && !strncmp (argv[from], standard_args[i].longname,
1431 thislen))
1432 {
1433 if (match == -1)
1434 match = i;
1435 else
1436 match = -2;
1437 }
1438
1439 /* If we found exactly one match, use that. */
1440 if (match >= 0)
1441 {
1442 options[from] = standard_args[match].nargs;
1443 priority[from] = standard_args[match].priority;
1444 /* If --OPTION=VALUE syntax is used,
1445 this option uses just one argv element. */
1446 if (equals != 0)
1447 options[from] = 0;
1448 if (from + options[from] >= argc)
1449 fatal ("Option `%s' requires an argument\n", argv[from]);
1450 from += options[from];
1451 }
1452 }
1453 done: ;
1454 }
1455 }
1456
1457 /* Copy the arguments, in order of decreasing priority, to NEW. */
1458 new[0] = argv[0];
1459 while (to < argc)
1460 {
1461 int best = -1;
1462 int best_priority = -9999;
1463
1464 /* Find the highest priority remaining option.
1465 If several have equal priority, take the first of them. */
1466 for (from = 1; from < argc; from++)
1467 {
1468 if (argv[from] != 0 && priority[from] > best_priority)
1469 {
1470 best_priority = priority[from];
1471 best = from;
1472 }
1473 /* Skip option arguments--they are tied to the options. */
1474 if (options[from] > 0)
1475 from += options[from];
1476 }
1477
1478 if (best < 0)
1479 abort ();
1480
1481 /* Copy the highest priority remaining option, with its args, to NEW. */
1482 new[to++] = argv[best];
1483 for (i = 0; i < options[best]; i++)
1484 new[to++] = argv[best + i + 1];
1485
1486 /* Clear out this option in ARGV. */
1487 argv[best] = 0;
1488 for (i = 0; i < options[best]; i++)
1489 argv[best + i + 1] = 0;
1490 }
1491
1492 bcopy (new, argv, sizeof (char *) * argc);
1493
1494 free (options);
1495 free (new);
1496 free (priority);
1497 }
1498 \f
1499 DEFUN ("kill-emacs", Fkill_emacs, Skill_emacs, 0, 1, "P",
1500 "Exit the Emacs job and kill it.\n\
1501 If ARG is an integer, return ARG as the exit program code.\n\
1502 If ARG is a string, stuff it as keyboard input.\n\n\
1503 The value of `kill-emacs-hook', if not void,\n\
1504 is a list of functions (of no args),\n\
1505 all of which are called before Emacs is actually killed.")
1506 (arg)
1507 Lisp_Object arg;
1508 {
1509 Lisp_Object hook, hook1;
1510 int i;
1511 struct gcpro gcpro1;
1512
1513 GCPRO1 (arg);
1514
1515 if (feof (stdin))
1516 arg = Qt;
1517
1518 if (!NILP (Vrun_hooks) && !noninteractive)
1519 call1 (Vrun_hooks, intern ("kill-emacs-hook"));
1520
1521 UNGCPRO;
1522
1523 /* Is it really necessary to do this deassign
1524 when we are going to exit anyway? */
1525 /* #ifdef VMS
1526 stop_vms_input ();
1527 #endif */
1528
1529 shut_down_emacs (0, 0, STRINGP (arg) ? arg : Qnil);
1530
1531 /* If we have an auto-save list file,
1532 kill it because we are exiting Emacs deliberately (not crashing).
1533 Do it after shut_down_emacs, which does an auto-save. */
1534 if (STRINGP (Vauto_save_list_file_name))
1535 unlink (XSTRING (Vauto_save_list_file_name)->data);
1536
1537 exit (INTEGERP (arg) ? XINT (arg)
1538 #ifdef VMS
1539 : 1
1540 #else
1541 : 0
1542 #endif
1543 );
1544 /* NOTREACHED */
1545 }
1546
1547
1548 /* Perform an orderly shutdown of Emacs. Autosave any modified
1549 buffers, kill any child processes, clean up the terminal modes (if
1550 we're in the foreground), and other stuff like that. Don't perform
1551 any redisplay; this may be called when Emacs is shutting down in
1552 the background, or after its X connection has died.
1553
1554 If SIG is a signal number, print a message for it.
1555
1556 This is called by fatal signal handlers, X protocol error handlers,
1557 and Fkill_emacs. */
1558
1559 void
1560 shut_down_emacs (sig, no_x, stuff)
1561 int sig, no_x;
1562 Lisp_Object stuff;
1563 {
1564 /* Prevent running of hooks from now on. */
1565 Vrun_hooks = Qnil;
1566
1567 /* If we are controlling the terminal, reset terminal modes */
1568 #ifdef EMACS_HAVE_TTY_PGRP
1569 {
1570 int pgrp = EMACS_GETPGRP (0);
1571
1572 int tpgrp;
1573 if (EMACS_GET_TTY_PGRP (0, &tpgrp) != -1
1574 && tpgrp == pgrp)
1575 {
1576 fflush (stdout);
1577 reset_sys_modes ();
1578 if (sig && sig != SIGTERM)
1579 fprintf (stderr, "Fatal error (%d).", sig);
1580 }
1581 }
1582 #else
1583 fflush (stdout);
1584 reset_sys_modes ();
1585 #endif
1586
1587 stuff_buffered_input (stuff);
1588
1589 kill_buffer_processes (Qnil);
1590 Fdo_auto_save (Qt, Qnil);
1591
1592 #ifdef CLASH_DETECTION
1593 unlock_all_files ();
1594 #endif
1595
1596 #ifdef VMS
1597 kill_vms_processes ();
1598 #endif
1599
1600 #if 0 /* This triggers a bug in XCloseDisplay and is not needed. */
1601 #ifdef HAVE_X_WINDOWS
1602 /* It's not safe to call intern here. Maybe we are crashing. */
1603 if (!noninteractive && SYMBOLP (Vwindow_system)
1604 && XSYMBOL (Vwindow_system)->name->size == 1
1605 && XSYMBOL (Vwindow_system)->name->data[0] == 'x'
1606 && ! no_x)
1607 Fx_close_current_connection ();
1608 #endif /* HAVE_X_WINDOWS */
1609 #endif
1610
1611 #ifdef SIGIO
1612 /* There is a tendency for a SIGIO signal to arrive within exit,
1613 and cause a SIGHUP because the input descriptor is already closed. */
1614 unrequest_sigio ();
1615 signal (SIGIO, SIG_IGN);
1616 #endif
1617
1618 #ifdef WINDOWSNT
1619 term_ntproc ();
1620 #endif
1621
1622 #ifdef MSDOS
1623 dos_cleanup ();
1624 #endif
1625 }
1626
1627
1628 \f
1629 #ifndef CANNOT_DUMP
1630
1631 #ifdef HAVE_SHM
1632
1633 DEFUN ("dump-emacs-data", Fdump_emacs_data, Sdump_emacs_data, 1, 1, 0,
1634 "Dump current state of Emacs into data file FILENAME.\n\
1635 This function exists on systems that use HAVE_SHM.")
1636 (filename)
1637 Lisp_Object filename;
1638 {
1639 extern char my_edata[];
1640 Lisp_Object tem;
1641
1642 CHECK_STRING (filename, 0);
1643 filename = Fexpand_file_name (filename, Qnil);
1644
1645 tem = Vpurify_flag;
1646 Vpurify_flag = Qnil;
1647
1648 fflush (stdout);
1649 /* Tell malloc where start of impure now is */
1650 /* Also arrange for warnings when nearly out of space. */
1651 #ifndef SYSTEM_MALLOC
1652 memory_warnings (my_edata, malloc_warning);
1653 #endif
1654 map_out_data (XSTRING (filename)->data);
1655
1656 Vpurify_flag = tem;
1657
1658 return Qnil;
1659 }
1660
1661 #else /* not HAVE_SHM */
1662
1663 DEFUN ("dump-emacs", Fdump_emacs, Sdump_emacs, 2, 2, 0,
1664 "Dump current state of Emacs into executable file FILENAME.\n\
1665 Take symbols from SYMFILE (presumably the file you executed to run Emacs).\n\
1666 This is used in the file `loadup.el' when building Emacs.\n\
1667 \n\
1668 You must run Emacs in batch mode in order to dump it.")
1669 (filename, symfile)
1670 Lisp_Object filename, symfile;
1671 {
1672 extern char my_edata[];
1673 Lisp_Object tem;
1674 Lisp_Object symbol;
1675 int count = specpdl_ptr - specpdl;
1676
1677 if (! noninteractive)
1678 error ("Dumping Emacs works only in batch mode");
1679
1680 /* Bind `command-line-processed' to nil before dumping,
1681 so that the dumped Emacs will process its command line
1682 and set up to work with X windows if appropriate. */
1683 symbol = intern ("command-line-process");
1684 specbind (symbol, Qnil);
1685
1686 CHECK_STRING (filename, 0);
1687 filename = Fexpand_file_name (filename, Qnil);
1688 if (!NILP (symfile))
1689 {
1690 CHECK_STRING (symfile, 0);
1691 if (XSTRING (symfile)->size)
1692 symfile = Fexpand_file_name (symfile, Qnil);
1693 }
1694
1695 tem = Vpurify_flag;
1696 Vpurify_flag = Qnil;
1697
1698 #ifdef HAVE_TZSET
1699 set_time_zone_rule (dump_tz);
1700 #ifndef LOCALTIME_CACHE
1701 /* Force a tz reload, since set_time_zone_rule doesn't. */
1702 tzset ();
1703 #endif
1704 #endif
1705
1706 fflush (stdout);
1707 #ifdef VMS
1708 mapout_data (XSTRING (filename)->data);
1709 #else
1710 /* Tell malloc where start of impure now is */
1711 /* Also arrange for warnings when nearly out of space. */
1712 #ifndef SYSTEM_MALLOC
1713 #ifndef WINDOWSNT
1714 /* On Windows, this was done before dumping, and that once suffices.
1715 Meanwhile, my_edata is not valid on Windows. */
1716 memory_warnings (my_edata, malloc_warning);
1717 #endif /* not WINDOWSNT */
1718 #endif
1719 #ifdef DOUG_LEA_MALLOC
1720 malloc_state_ptr = malloc_get_state ();
1721 #endif
1722 unexec (XSTRING (filename)->data,
1723 !NILP (symfile) ? XSTRING (symfile)->data : 0, my_edata, 0, 0);
1724 #ifdef DOUG_LEA_MALLOC
1725 free (malloc_state_ptr);
1726 #endif
1727 #endif /* not VMS */
1728
1729 Vpurify_flag = tem;
1730
1731 return unbind_to (count, Qnil);
1732 }
1733
1734 #endif /* not HAVE_SHM */
1735
1736 #endif /* not CANNOT_DUMP */
1737 \f
1738 #ifndef SEPCHAR
1739 #define SEPCHAR ':'
1740 #endif
1741
1742 Lisp_Object
1743 decode_env_path (evarname, defalt)
1744 char *evarname, *defalt;
1745 {
1746 register char *path, *p;
1747 Lisp_Object lpath, element, tem;
1748
1749 /* It's okay to use getenv here, because this function is only used
1750 to initialize variables when Emacs starts up, and isn't called
1751 after that. */
1752 if (evarname != 0)
1753 path = (char *) getenv (evarname);
1754 else
1755 path = 0;
1756 if (!path)
1757 path = defalt;
1758 #ifdef DOS_NT
1759 /* Ensure values from the environment use the proper directory separator. */
1760 if (path)
1761 {
1762 p = alloca (strlen (path) + 1);
1763 strcpy (p, path);
1764 path = p;
1765
1766 if ('/' == DIRECTORY_SEP)
1767 dostounix_filename (path);
1768 else
1769 unixtodos_filename (path);
1770 }
1771 #endif
1772 lpath = Qnil;
1773 while (1)
1774 {
1775 p = index (path, SEPCHAR);
1776 if (!p) p = path + strlen (path);
1777 element = (p - path ? make_string (path, p - path)
1778 : build_string ("."));
1779
1780 /* Add /: to the front of the name
1781 if it would otherwise be treated as magic. */
1782 tem = Ffind_file_name_handler (element, Qt);
1783 if (! NILP (tem))
1784 element = concat2 (build_string ("/:"), element);
1785
1786 lpath = Fcons (element, lpath);
1787 if (*p)
1788 path = p + 1;
1789 else
1790 break;
1791 }
1792 return Fnreverse (lpath);
1793 }
1794
1795 void
1796 syms_of_emacs ()
1797 {
1798 Qfile_name_handler_alist = intern ("file-name-handler-alist");
1799 staticpro (&Qfile_name_handler_alist);
1800
1801 #ifndef CANNOT_DUMP
1802 #ifdef HAVE_SHM
1803 defsubr (&Sdump_emacs_data);
1804 #else
1805 defsubr (&Sdump_emacs);
1806 #endif
1807 #endif
1808
1809 defsubr (&Skill_emacs);
1810
1811 defsubr (&Sinvocation_name);
1812 defsubr (&Sinvocation_directory);
1813
1814 DEFVAR_LISP ("command-line-args", &Vcommand_line_args,
1815 "Args passed by shell to Emacs, as a list of strings.");
1816
1817 DEFVAR_LISP ("system-type", &Vsystem_type,
1818 "Value is symbol indicating type of operating system you are using.");
1819 Vsystem_type = intern (SYSTEM_TYPE);
1820
1821 DEFVAR_LISP ("system-configuration", &Vsystem_configuration,
1822 "Value is string indicating configuration Emacs was built for.");
1823 Vsystem_configuration = build_string (EMACS_CONFIGURATION);
1824
1825 DEFVAR_LISP ("system-configuration-options", &Vsystem_configuration_options,
1826 "String containing the configuration options Emacs was built with.");
1827 Vsystem_configuration_options = build_string (EMACS_CONFIG_OPTIONS);
1828
1829 DEFVAR_BOOL ("noninteractive", &noninteractive1,
1830 "Non-nil means Emacs is running without interactive terminal.");
1831
1832 DEFVAR_LISP ("kill-emacs-hook", &Vkill_emacs_hook,
1833 "Hook to be run whenever kill-emacs is called.\n\
1834 Since kill-emacs may be invoked when the terminal is disconnected (or\n\
1835 in other similar situations), functions placed on this hook should not\n\
1836 expect to be able to interact with the user. To ask for confirmation,\n\
1837 see `kill-emacs-query-functions' instead.");
1838 Vkill_emacs_hook = Qnil;
1839
1840 #ifdef SIGUSR1
1841 DEFVAR_LISP ("signal-USR1-hook", &Vsignal_USR1_hook,
1842 "Hook to be run whenever emacs receives a USR1 signal");
1843 Vsignal_USR1_hook = Qnil;
1844 #ifdef SIGUSR2
1845 DEFVAR_LISP ("signal-USR2-hook", &Vsignal_USR2_hook,
1846 "Hook to be run whenever emacs receives a USR2 signal");
1847 Vsignal_USR2_hook = Qnil;
1848 #endif
1849 #endif
1850
1851
1852 DEFVAR_INT ("emacs-priority", &emacs_priority,
1853 "Priority for Emacs to run at.\n\
1854 This value is effective only if set before Emacs is dumped,\n\
1855 and only if the Emacs executable is installed with setuid to permit\n\
1856 it to change priority. (Emacs sets its uid back to the real uid.)\n\
1857 Currently, you need to define SET_EMACS_PRIORITY in `config.h'\n\
1858 before you compile Emacs, to enable the code for this feature.");
1859 emacs_priority = 0;
1860
1861 DEFVAR_LISP ("path-separator", &Vpath_separator,
1862 "The directory separator in search paths, as a string.");
1863 {
1864 char c = SEPCHAR;
1865 Vpath_separator = make_string (&c, 1);
1866 }
1867
1868 DEFVAR_LISP ("invocation-name", &Vinvocation_name,
1869 "The program name that was used to run Emacs.\n\
1870 Any directory names are omitted.");
1871
1872 DEFVAR_LISP ("invocation-directory", &Vinvocation_directory,
1873 "The directory in which the Emacs executable was found, to run it.\n\
1874 The value is nil if that directory's name is not known.");
1875
1876 DEFVAR_LISP ("installation-directory", &Vinstallation_directory,
1877 "A directory within which to look for the `lib-src' and `etc' directories.\n\
1878 This is non-nil when we can't find those directories in their standard\n\
1879 installed locations, but we can find them\n\
1880 near where the Emacs executable was found.");
1881 Vinstallation_directory = Qnil;
1882 }