]> code.delx.au - gnu-emacs/blob - src/emacs.c
(Fnext_property_change): Properly offset interval
[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 /* Intern the names of all standard functions and variables;
1080 define standard keys. */
1081
1082 if (!initialized)
1083 {
1084 /* The basic levels of Lisp must come first */
1085 /* And data must come first of all
1086 for the sake of symbols like error-message */
1087 syms_of_data ();
1088 syms_of_alloc ();
1089 syms_of_lread ();
1090 syms_of_print ();
1091 syms_of_eval ();
1092 syms_of_fns ();
1093 syms_of_floatfns ();
1094
1095 syms_of_abbrev ();
1096 syms_of_buffer ();
1097 syms_of_bytecode ();
1098 syms_of_callint ();
1099 syms_of_casefiddle ();
1100 syms_of_casetab ();
1101 syms_of_callproc ();
1102 syms_of_category ();
1103 syms_of_ccl ();
1104 syms_of_charset ();
1105 syms_of_cmds ();
1106 #ifndef NO_DIR_LIBRARY
1107 syms_of_dired ();
1108 #endif /* not NO_DIR_LIBRARY */
1109 syms_of_display ();
1110 syms_of_doc ();
1111 syms_of_editfns ();
1112 syms_of_emacs ();
1113 syms_of_fileio ();
1114 syms_of_coding (); /* This should be after syms_of_fileio. */
1115 #ifdef CLASH_DETECTION
1116 syms_of_filelock ();
1117 #endif /* CLASH_DETECTION */
1118 syms_of_indent ();
1119 syms_of_insdel ();
1120 syms_of_keyboard ();
1121 syms_of_keymap ();
1122 syms_of_macros ();
1123 syms_of_marker ();
1124 syms_of_minibuf ();
1125 syms_of_mocklisp ();
1126 syms_of_process ();
1127 syms_of_search ();
1128 syms_of_frame ();
1129 syms_of_syntax ();
1130 syms_of_term ();
1131 syms_of_undo ();
1132
1133 /* Only defined if Emacs is compiled with USE_TEXT_PROPERTIES */
1134 syms_of_textprop ();
1135 #ifdef VMS
1136 syms_of_vmsproc ();
1137 #endif /* VMS */
1138 #ifdef WINDOWSNT
1139 syms_of_ntproc ();
1140 #endif /* WINDOWSNT */
1141 syms_of_window ();
1142 syms_of_xdisp ();
1143 #ifdef HAVE_X_WINDOWS
1144 syms_of_xterm ();
1145 syms_of_xfns ();
1146 syms_of_fontset ();
1147 #ifdef HAVE_X11
1148 syms_of_xselect ();
1149 #endif
1150 #endif /* HAVE_X_WINDOWS */
1151
1152 #ifndef HAVE_NTGUI
1153 syms_of_xfaces ();
1154 syms_of_xmenu ();
1155 #endif
1156
1157 #ifdef HAVE_NTGUI
1158 syms_of_w32term ();
1159 syms_of_w32fns ();
1160 syms_of_w32faces ();
1161 syms_of_w32select ();
1162 syms_of_w32menu ();
1163 #endif /* HAVE_NTGUI */
1164
1165 #ifdef SYMS_SYSTEM
1166 SYMS_SYSTEM;
1167 #endif
1168
1169 #ifdef SYMS_MACHINE
1170 SYMS_MACHINE;
1171 #endif
1172
1173 keys_of_casefiddle ();
1174 keys_of_cmds ();
1175 keys_of_buffer ();
1176 keys_of_keyboard ();
1177 keys_of_keymap ();
1178 keys_of_macros ();
1179 keys_of_minibuf ();
1180 keys_of_window ();
1181 keys_of_frame ();
1182 }
1183
1184 if (!noninteractive)
1185 {
1186 #ifdef VMS
1187 init_vms_input ();/* init_display calls get_frame_size, that needs this */
1188 #endif /* VMS */
1189 init_display (); /* Determine terminal type. init_sys_modes uses results */
1190 }
1191 init_keyboard (); /* This too must precede init_sys_modes */
1192 #ifdef VMS
1193 init_vmsproc (); /* And this too. */
1194 #endif /* VMS */
1195 init_sys_modes (); /* Init system terminal modes (RAW or CBREAK, etc.) */
1196 init_xdisp ();
1197 init_macros ();
1198 init_editfns ();
1199 #ifdef LISP_FLOAT_TYPE
1200 init_floatfns ();
1201 #endif
1202 #ifdef VMS
1203 init_vmsfns ();
1204 #endif /* VMS */
1205 init_process ();
1206
1207 if (!initialized)
1208 {
1209 char *file;
1210 /* Handle -l loadup-and-dump, args passed by Makefile. */
1211 if (argmatch (argv, argc, "-l", "--load", 3, &file, &skip_args))
1212 Vtop_level = Fcons (intern ("load"),
1213 Fcons (build_string (file), Qnil));
1214 #ifdef CANNOT_DUMP
1215 /* Unless next switch is -nl, load "loadup.el" first thing. */
1216 if (!argmatch (argv, argc, "-nl", "--no-loadup", 6, NULL, &skip_args))
1217 Vtop_level = Fcons (intern ("load"),
1218 Fcons (build_string ("loadup.el"), Qnil));
1219 #endif /* CANNOT_DUMP */
1220 }
1221
1222 if (initialized)
1223 {
1224 #ifdef HAVE_TZSET
1225 {
1226 /* If the execution TZ happens to be the same as the dump TZ,
1227 change it to some other value and then change it back,
1228 to force the underlying implementation to reload the TZ info.
1229 This is needed on implementations that load TZ info from files,
1230 since the TZ file contents may differ between dump and execution. */
1231 char *tz = getenv ("TZ");
1232 if (tz && !strcmp (tz, dump_tz))
1233 {
1234 ++*tz;
1235 tzset ();
1236 --*tz;
1237 }
1238 }
1239 #endif
1240 }
1241
1242 /* Gerd Moellmann <gerd@acm.org> says this makes profiling work on
1243 FreeBSD. It might work on some other systems too.
1244 Give it a try and tell me if it works on your system. */
1245 #ifdef __FreeBSD__
1246 #ifdef PROFILING
1247 if (initialized)
1248 {
1249 extern void _mcleanup ();
1250 extern char etext;
1251 extern Lisp_Object Fredraw_frame ();
1252 atexit (_mcleanup);
1253 /* This uses Fredraw_frame because that function
1254 comes first in the Emacs executable.
1255 It might be better to use something that gives
1256 the start of the text segment, but start_of_text
1257 is not defined on all systems now. */
1258 monstartup (Fredraw_frame, &etext);
1259 }
1260 else
1261 moncontrol (0);
1262 #endif
1263 #endif
1264
1265 initialized = 1;
1266
1267 #ifdef LOCALTIME_CACHE
1268 /* Some versions of localtime have a bug. They cache the value of the time
1269 zone rather than looking it up every time. Since localtime() is
1270 called to bolt the undumping time into the undumped emacs, this
1271 results in localtime ignoring the TZ environment variable.
1272 This flushes the new TZ value into localtime. */
1273 tzset ();
1274 #endif /* defined (LOCALTIME_CACHE) */
1275
1276 /* Enter editor command loop. This never returns. */
1277 Frecursive_edit ();
1278 /* NOTREACHED */
1279 }
1280 \f
1281 /* Sort the args so we can find the most important ones
1282 at the beginning of argv. */
1283
1284 /* First, here's a table of all the standard options. */
1285
1286 struct standard_args
1287 {
1288 char *name;
1289 char *longname;
1290 int priority;
1291 int nargs;
1292 };
1293
1294 struct standard_args standard_args[] =
1295 {
1296 { "-version", "--version", 110, 0 },
1297 { "-help", "--help", 110, 0 },
1298 { "-nl", "--no-shared-memory", 100, 0 },
1299 #ifdef VMS
1300 { "-map", "--map-data", 100, 0 },
1301 #endif
1302 { "-t", "--terminal", 90, 1 },
1303 { "-d", "--display", 80, 1 },
1304 { "-display", 0, 80, 1 },
1305 { "-nw", "--no-windows", 70, 0 },
1306 { "-no-unibyte", "--no-unibyte", 66, 0 },
1307 { "-multibyte", "--multibyte", 66, 0 },
1308 { "-unibyte", "--unibyte", 65, 0 },
1309 { "-no-multibyte", "--no-multibyte", 65, 0 },
1310 { "-batch", "--batch", 60, 0 },
1311 { "-q", "--no-init-file", 50, 0 },
1312 { "-no-init-file", 0, 50, 0 },
1313 { "-no-site-file", "--no-site-file", 40, 0 },
1314 { "-u", "--user", 30, 1 },
1315 { "-user", 0, 30, 1 },
1316 { "-debug-init", "--debug-init", 20, 0 },
1317 { "-i", "--icon-type", 15, 0 },
1318 { "-itype", 0, 15, 0 },
1319 { "-iconic", "--iconic", 15, 0 },
1320 { "-bg", "--background-color", 10, 1 },
1321 { "-background", 0, 10, 1 },
1322 { "-fg", "--foreground-color", 10, 1 },
1323 { "-foreground", 0, 10, 1 },
1324 { "-bd", "--border-color", 10, 1 },
1325 { "-bw", "--border-width", 10, 1 },
1326 { "-ib", "--internal-border", 10, 1 },
1327 { "-ms", "--mouse-color", 10, 1 },
1328 { "-cr", "--cursor-color", 10, 1 },
1329 { "-fn", "--font", 10, 1 },
1330 { "-font", 0, 10, 1 },
1331 { "-g", "--geometry", 10, 1 },
1332 { "-geometry", 0, 10, 1 },
1333 { "-T", "--title", 10, 1 },
1334 { "-title", 0, 10, 1 },
1335 { "-name", "--name", 10, 1 },
1336 { "-xrm", "--xrm", 10, 1 },
1337 { "-r", "--reverse-video", 5, 0 },
1338 { "-rv", 0, 5, 0 },
1339 { "-reverse", 0, 5, 0 },
1340 { "-hb", "--horizontal-scroll-bars", 5, 0 },
1341 { "-vb", "--vertical-scroll-bars", 5, 0 },
1342 /* These have the same priority as ordinary file name args,
1343 so they are not reordered with respect to those. */
1344 { "-L", "--directory", 0, 1 },
1345 { "-directory", 0, 0, 1 },
1346 { "-l", "--load", 0, 1 },
1347 { "-load", 0, 0, 1 },
1348 { "-f", "--funcall", 0, 1 },
1349 { "-funcall", 0, 0, 1 },
1350 { "-eval", "--eval", 0, 1 },
1351 { "-find-file", "--find-file", 0, 1 },
1352 { "-visit", "--visit", 0, 1 },
1353 { "-insert", "--insert", 0, 1 },
1354 /* This should be processed after ordinary file name args and the like. */
1355 { "-kill", "--kill", -10, 0 },
1356 };
1357
1358 /* Reorder the elements of ARGV (assumed to have ARGC elements)
1359 so that the highest priority ones come first.
1360 Do not change the order of elements of equal priority.
1361 If an option takes an argument, keep it and its argument together. */
1362
1363 static void
1364 sort_args (argc, argv)
1365 int argc;
1366 char **argv;
1367 {
1368 char **new = (char **) xmalloc (sizeof (char *) * argc);
1369 /* For each element of argv,
1370 the corresponding element of options is:
1371 0 for an option that takes no arguments,
1372 1 for an option that takes one argument, etc.
1373 -1 for an ordinary non-option argument. */
1374 int *options = (int *) xmalloc (sizeof (int) * argc);
1375 int *priority = (int *) xmalloc (sizeof (int) * argc);
1376 int to = 1;
1377 int from;
1378 int i;
1379 int end_of_options = argc;
1380
1381 /* Categorize all the options,
1382 and figure out which argv elts are option arguments. */
1383 for (from = 1; from < argc; from++)
1384 {
1385 options[from] = -1;
1386 priority[from] = 0;
1387 if (argv[from][0] == '-')
1388 {
1389 int match, thislen;
1390 char *equals;
1391
1392 /* If we have found "--", don't consider
1393 any more arguments as options. */
1394 if (argv[from][1] == '-' && argv[from][2] == 0)
1395 {
1396 /* Leave the "--", and everything following it, at the end. */
1397 for (; from < argc; from++)
1398 {
1399 priority[from] = -100;
1400 options[from] = -1;
1401 }
1402 break;
1403 }
1404
1405 /* Look for a match with a known old-fashioned option. */
1406 for (i = 0; i < sizeof (standard_args) / sizeof (standard_args[0]); i++)
1407 if (!strcmp (argv[from], standard_args[i].name))
1408 {
1409 options[from] = standard_args[i].nargs;
1410 priority[from] = standard_args[i].priority;
1411 if (from + standard_args[i].nargs >= argc)
1412 fatal ("Option `%s' requires an argument\n", argv[from]);
1413 from += standard_args[i].nargs;
1414 goto done;
1415 }
1416
1417 /* Look for a match with a known long option.
1418 MATCH is -1 if no match so far, -2 if two or more matches so far,
1419 >= 0 (the table index of the match) if just one match so far. */
1420 if (argv[from][1] == '-')
1421 {
1422 match = -1;
1423 thislen = strlen (argv[from]);
1424 equals = index (argv[from], '=');
1425 if (equals != 0)
1426 thislen = equals - argv[from];
1427
1428 for (i = 0;
1429 i < sizeof (standard_args) / sizeof (standard_args[0]); i++)
1430 if (standard_args[i].longname
1431 && !strncmp (argv[from], standard_args[i].longname,
1432 thislen))
1433 {
1434 if (match == -1)
1435 match = i;
1436 else
1437 match = -2;
1438 }
1439
1440 /* If we found exactly one match, use that. */
1441 if (match >= 0)
1442 {
1443 options[from] = standard_args[match].nargs;
1444 priority[from] = standard_args[match].priority;
1445 /* If --OPTION=VALUE syntax is used,
1446 this option uses just one argv element. */
1447 if (equals != 0)
1448 options[from] = 0;
1449 if (from + options[from] >= argc)
1450 fatal ("Option `%s' requires an argument\n", argv[from]);
1451 from += options[from];
1452 }
1453 }
1454 done: ;
1455 }
1456 }
1457
1458 /* Copy the arguments, in order of decreasing priority, to NEW. */
1459 new[0] = argv[0];
1460 while (to < argc)
1461 {
1462 int best = -1;
1463 int best_priority = -9999;
1464
1465 /* Find the highest priority remaining option.
1466 If several have equal priority, take the first of them. */
1467 for (from = 1; from < argc; from++)
1468 {
1469 if (argv[from] != 0 && priority[from] > best_priority)
1470 {
1471 best_priority = priority[from];
1472 best = from;
1473 }
1474 /* Skip option arguments--they are tied to the options. */
1475 if (options[from] > 0)
1476 from += options[from];
1477 }
1478
1479 if (best < 0)
1480 abort ();
1481
1482 /* Copy the highest priority remaining option, with its args, to NEW. */
1483 new[to++] = argv[best];
1484 for (i = 0; i < options[best]; i++)
1485 new[to++] = argv[best + i + 1];
1486
1487 /* Clear out this option in ARGV. */
1488 argv[best] = 0;
1489 for (i = 0; i < options[best]; i++)
1490 argv[best + i + 1] = 0;
1491 }
1492
1493 bcopy (new, argv, sizeof (char *) * argc);
1494
1495 free (options);
1496 free (new);
1497 free (priority);
1498 }
1499 \f
1500 DEFUN ("kill-emacs", Fkill_emacs, Skill_emacs, 0, 1, "P",
1501 "Exit the Emacs job and kill it.\n\
1502 If ARG is an integer, return ARG as the exit program code.\n\
1503 If ARG is a string, stuff it as keyboard input.\n\n\
1504 The value of `kill-emacs-hook', if not void,\n\
1505 is a list of functions (of no args),\n\
1506 all of which are called before Emacs is actually killed.")
1507 (arg)
1508 Lisp_Object arg;
1509 {
1510 Lisp_Object hook, hook1;
1511 int i;
1512 struct gcpro gcpro1;
1513
1514 GCPRO1 (arg);
1515
1516 if (feof (stdin))
1517 arg = Qt;
1518
1519 if (!NILP (Vrun_hooks) && !noninteractive)
1520 call1 (Vrun_hooks, intern ("kill-emacs-hook"));
1521
1522 UNGCPRO;
1523
1524 /* Is it really necessary to do this deassign
1525 when we are going to exit anyway? */
1526 /* #ifdef VMS
1527 stop_vms_input ();
1528 #endif */
1529
1530 shut_down_emacs (0, 0, STRINGP (arg) ? arg : Qnil);
1531
1532 /* If we have an auto-save list file,
1533 kill it because we are exiting Emacs deliberately (not crashing).
1534 Do it after shut_down_emacs, which does an auto-save. */
1535 if (STRINGP (Vauto_save_list_file_name))
1536 unlink (XSTRING (Vauto_save_list_file_name)->data);
1537
1538 exit (INTEGERP (arg) ? XINT (arg)
1539 #ifdef VMS
1540 : 1
1541 #else
1542 : 0
1543 #endif
1544 );
1545 /* NOTREACHED */
1546 }
1547
1548
1549 /* Perform an orderly shutdown of Emacs. Autosave any modified
1550 buffers, kill any child processes, clean up the terminal modes (if
1551 we're in the foreground), and other stuff like that. Don't perform
1552 any redisplay; this may be called when Emacs is shutting down in
1553 the background, or after its X connection has died.
1554
1555 If SIG is a signal number, print a message for it.
1556
1557 This is called by fatal signal handlers, X protocol error handlers,
1558 and Fkill_emacs. */
1559
1560 void
1561 shut_down_emacs (sig, no_x, stuff)
1562 int sig, no_x;
1563 Lisp_Object stuff;
1564 {
1565 /* Prevent running of hooks from now on. */
1566 Vrun_hooks = Qnil;
1567
1568 /* If we are controlling the terminal, reset terminal modes */
1569 #ifdef EMACS_HAVE_TTY_PGRP
1570 {
1571 int pgrp = EMACS_GETPGRP (0);
1572
1573 int tpgrp;
1574 if (EMACS_GET_TTY_PGRP (0, &tpgrp) != -1
1575 && tpgrp == pgrp)
1576 {
1577 fflush (stdout);
1578 reset_sys_modes ();
1579 if (sig && sig != SIGTERM)
1580 fprintf (stderr, "Fatal error (%d).", sig);
1581 }
1582 }
1583 #else
1584 fflush (stdout);
1585 reset_sys_modes ();
1586 #endif
1587
1588 stuff_buffered_input (stuff);
1589
1590 kill_buffer_processes (Qnil);
1591 Fdo_auto_save (Qt, Qnil);
1592
1593 #ifdef CLASH_DETECTION
1594 unlock_all_files ();
1595 #endif
1596
1597 #ifdef VMS
1598 kill_vms_processes ();
1599 #endif
1600
1601 #if 0 /* This triggers a bug in XCloseDisplay and is not needed. */
1602 #ifdef HAVE_X_WINDOWS
1603 /* It's not safe to call intern here. Maybe we are crashing. */
1604 if (!noninteractive && SYMBOLP (Vwindow_system)
1605 && XSYMBOL (Vwindow_system)->name->size == 1
1606 && XSYMBOL (Vwindow_system)->name->data[0] == 'x'
1607 && ! no_x)
1608 Fx_close_current_connection ();
1609 #endif /* HAVE_X_WINDOWS */
1610 #endif
1611
1612 #ifdef SIGIO
1613 /* There is a tendency for a SIGIO signal to arrive within exit,
1614 and cause a SIGHUP because the input descriptor is already closed. */
1615 unrequest_sigio ();
1616 signal (SIGIO, SIG_IGN);
1617 #endif
1618
1619 #ifdef WINDOWSNT
1620 term_ntproc ();
1621 #endif
1622
1623 #ifdef MSDOS
1624 dos_cleanup ();
1625 #endif
1626 }
1627
1628
1629 \f
1630 #ifndef CANNOT_DUMP
1631
1632 #ifdef HAVE_SHM
1633
1634 DEFUN ("dump-emacs-data", Fdump_emacs_data, Sdump_emacs_data, 1, 1, 0,
1635 "Dump current state of Emacs into data file FILENAME.\n\
1636 This function exists on systems that use HAVE_SHM.")
1637 (filename)
1638 Lisp_Object filename;
1639 {
1640 extern char my_edata[];
1641 Lisp_Object tem;
1642
1643 CHECK_STRING (filename, 0);
1644 filename = Fexpand_file_name (filename, Qnil);
1645
1646 tem = Vpurify_flag;
1647 Vpurify_flag = Qnil;
1648
1649 fflush (stdout);
1650 /* Tell malloc where start of impure now is */
1651 /* Also arrange for warnings when nearly out of space. */
1652 #ifndef SYSTEM_MALLOC
1653 memory_warnings (my_edata, malloc_warning);
1654 #endif
1655 map_out_data (XSTRING (filename)->data);
1656
1657 Vpurify_flag = tem;
1658
1659 return Qnil;
1660 }
1661
1662 #else /* not HAVE_SHM */
1663
1664 DEFUN ("dump-emacs", Fdump_emacs, Sdump_emacs, 2, 2, 0,
1665 "Dump current state of Emacs into executable file FILENAME.\n\
1666 Take symbols from SYMFILE (presumably the file you executed to run Emacs).\n\
1667 This is used in the file `loadup.el' when building Emacs.\n\
1668 \n\
1669 You must run Emacs in batch mode in order to dump it.")
1670 (filename, symfile)
1671 Lisp_Object filename, symfile;
1672 {
1673 extern char my_edata[];
1674 Lisp_Object tem;
1675 Lisp_Object symbol;
1676 int count = specpdl_ptr - specpdl;
1677
1678 if (! noninteractive)
1679 error ("Dumping Emacs works only in batch mode");
1680
1681 /* Bind `command-line-processed' to nil before dumping,
1682 so that the dumped Emacs will process its command line
1683 and set up to work with X windows if appropriate. */
1684 symbol = intern ("command-line-process");
1685 specbind (symbol, Qnil);
1686
1687 CHECK_STRING (filename, 0);
1688 filename = Fexpand_file_name (filename, Qnil);
1689 if (!NILP (symfile))
1690 {
1691 CHECK_STRING (symfile, 0);
1692 if (XSTRING (symfile)->size)
1693 symfile = Fexpand_file_name (symfile, Qnil);
1694 }
1695
1696 tem = Vpurify_flag;
1697 Vpurify_flag = Qnil;
1698
1699 #ifdef HAVE_TZSET
1700 set_time_zone_rule (dump_tz);
1701 #ifndef LOCALTIME_CACHE
1702 /* Force a tz reload, since set_time_zone_rule doesn't. */
1703 tzset ();
1704 #endif
1705 #endif
1706
1707 fflush (stdout);
1708 #ifdef VMS
1709 mapout_data (XSTRING (filename)->data);
1710 #else
1711 /* Tell malloc where start of impure now is */
1712 /* Also arrange for warnings when nearly out of space. */
1713 #ifndef SYSTEM_MALLOC
1714 #ifndef WINDOWSNT
1715 /* On Windows, this was done before dumping, and that once suffices.
1716 Meanwhile, my_edata is not valid on Windows. */
1717 memory_warnings (my_edata, malloc_warning);
1718 #endif /* not WINDOWSNT */
1719 #endif
1720 #ifdef DOUG_LEA_MALLOC
1721 malloc_state_ptr = malloc_get_state ();
1722 #endif
1723 unexec (XSTRING (filename)->data,
1724 !NILP (symfile) ? XSTRING (symfile)->data : 0, my_edata, 0, 0);
1725 #ifdef DOUG_LEA_MALLOC
1726 free (malloc_state_ptr);
1727 #endif
1728 #endif /* not VMS */
1729
1730 Vpurify_flag = tem;
1731
1732 return unbind_to (count, Qnil);
1733 }
1734
1735 #endif /* not HAVE_SHM */
1736
1737 #endif /* not CANNOT_DUMP */
1738 \f
1739 #ifndef SEPCHAR
1740 #define SEPCHAR ':'
1741 #endif
1742
1743 Lisp_Object
1744 decode_env_path (evarname, defalt)
1745 char *evarname, *defalt;
1746 {
1747 register char *path, *p;
1748 Lisp_Object lpath, element, tem;
1749
1750 /* It's okay to use getenv here, because this function is only used
1751 to initialize variables when Emacs starts up, and isn't called
1752 after that. */
1753 if (evarname != 0)
1754 path = (char *) getenv (evarname);
1755 else
1756 path = 0;
1757 if (!path)
1758 path = defalt;
1759 #ifdef DOS_NT
1760 /* Ensure values from the environment use the proper directory separator. */
1761 if (path)
1762 {
1763 p = alloca (strlen (path) + 1);
1764 strcpy (p, path);
1765 path = p;
1766
1767 if ('/' == DIRECTORY_SEP)
1768 dostounix_filename (path);
1769 else
1770 unixtodos_filename (path);
1771 }
1772 #endif
1773 lpath = Qnil;
1774 while (1)
1775 {
1776 p = index (path, SEPCHAR);
1777 if (!p) p = path + strlen (path);
1778 element = (p - path ? make_string (path, p - path)
1779 : build_string ("."));
1780
1781 /* Add /: to the front of the name
1782 if it would otherwise be treated as magic. */
1783 tem = Ffind_file_name_handler (element, Qt);
1784 if (! NILP (tem))
1785 element = concat2 (build_string ("/:"), element);
1786
1787 lpath = Fcons (element, lpath);
1788 if (*p)
1789 path = p + 1;
1790 else
1791 break;
1792 }
1793 return Fnreverse (lpath);
1794 }
1795
1796 void
1797 syms_of_emacs ()
1798 {
1799 Qfile_name_handler_alist = intern ("file-name-handler-alist");
1800 staticpro (&Qfile_name_handler_alist);
1801
1802 #ifndef CANNOT_DUMP
1803 #ifdef HAVE_SHM
1804 defsubr (&Sdump_emacs_data);
1805 #else
1806 defsubr (&Sdump_emacs);
1807 #endif
1808 #endif
1809
1810 defsubr (&Skill_emacs);
1811
1812 defsubr (&Sinvocation_name);
1813 defsubr (&Sinvocation_directory);
1814
1815 DEFVAR_LISP ("command-line-args", &Vcommand_line_args,
1816 "Args passed by shell to Emacs, as a list of strings.");
1817
1818 DEFVAR_LISP ("system-type", &Vsystem_type,
1819 "Value is symbol indicating type of operating system you are using.");
1820 Vsystem_type = intern (SYSTEM_TYPE);
1821
1822 DEFVAR_LISP ("system-configuration", &Vsystem_configuration,
1823 "Value is string indicating configuration Emacs was built for.");
1824 Vsystem_configuration = build_string (EMACS_CONFIGURATION);
1825
1826 DEFVAR_LISP ("system-configuration-options", &Vsystem_configuration_options,
1827 "String containing the configuration options Emacs was built with.");
1828 Vsystem_configuration_options = build_string (EMACS_CONFIG_OPTIONS);
1829
1830 DEFVAR_BOOL ("noninteractive", &noninteractive1,
1831 "Non-nil means Emacs is running without interactive terminal.");
1832
1833 DEFVAR_LISP ("kill-emacs-hook", &Vkill_emacs_hook,
1834 "Hook to be run whenever kill-emacs is called.\n\
1835 Since kill-emacs may be invoked when the terminal is disconnected (or\n\
1836 in other similar situations), functions placed on this hook should not\n\
1837 expect to be able to interact with the user. To ask for confirmation,\n\
1838 see `kill-emacs-query-functions' instead.");
1839 Vkill_emacs_hook = Qnil;
1840
1841 #ifdef SIGUSR1
1842 DEFVAR_LISP ("signal-USR1-hook", &Vsignal_USR1_hook,
1843 "Hook to be run whenever emacs receives a USR1 signal");
1844 Vsignal_USR1_hook = Qnil;
1845 #ifdef SIGUSR2
1846 DEFVAR_LISP ("signal-USR2-hook", &Vsignal_USR2_hook,
1847 "Hook to be run whenever emacs receives a USR2 signal");
1848 Vsignal_USR2_hook = Qnil;
1849 #endif
1850 #endif
1851
1852
1853 DEFVAR_INT ("emacs-priority", &emacs_priority,
1854 "Priority for Emacs to run at.\n\
1855 This value is effective only if set before Emacs is dumped,\n\
1856 and only if the Emacs executable is installed with setuid to permit\n\
1857 it to change priority. (Emacs sets its uid back to the real uid.)\n\
1858 Currently, you need to define SET_EMACS_PRIORITY in `config.h'\n\
1859 before you compile Emacs, to enable the code for this feature.");
1860 emacs_priority = 0;
1861
1862 DEFVAR_LISP ("path-separator", &Vpath_separator,
1863 "The directory separator in search paths, as a string.");
1864 {
1865 char c = SEPCHAR;
1866 Vpath_separator = make_string (&c, 1);
1867 }
1868
1869 DEFVAR_LISP ("invocation-name", &Vinvocation_name,
1870 "The program name that was used to run Emacs.\n\
1871 Any directory names are omitted.");
1872
1873 DEFVAR_LISP ("invocation-directory", &Vinvocation_directory,
1874 "The directory in which the Emacs executable was found, to run it.\n\
1875 The value is nil if that directory's name is not known.");
1876
1877 DEFVAR_LISP ("installation-directory", &Vinstallation_directory,
1878 "A directory within which to look for the `lib-src' and `etc' directories.\n\
1879 This is non-nil when we can't find those directories in their standard\n\
1880 installed locations, but we can find them\n\
1881 near where the Emacs executable was found.");
1882 Vinstallation_directory = Qnil;
1883 }