Mercurial > hg > xemacs-beta
view src/callproc.c @ 617:af57a77cbc92
[xemacs-hg @ 2001-06-18 07:09:50 by ben]
---------------------------------------------------------------
DOCUMENTATION FIXES:
---------------------------------------------------------------
eval.c: Correct documentation.
elhash.c: Doc correction.
---------------------------------------------------------------
LISP OBJECT CLEANUP:
---------------------------------------------------------------
bytecode.h, buffer.h, casetab.h, chartab.h, console-msw.h, console.h, database.c, device.h, eldap.h, elhash.h, events.h, extents.h, faces.h, file-coding.h, frame.h, glyphs.h, gui-x.h, gui.h, keymap.h, lisp-disunion.h, lisp-union.h, lisp.h, lrecord.h, lstream.h, mule-charset.h, objects.h, opaque.h, postgresql.h, process.h, rangetab.h, specifier.h, toolbar.h, tooltalk.h, ui-gtk.h: Add wrap_* to all objects (it was already there for a few of them)
-- an expression to encapsulate a pointer into a Lisp object,
rather than the inconvenient XSET*. "wrap" was chosen because
"make" as in make_int(), make_char() is not appropriate. (It
implies allocation. The issue does not exist for ints and chars
because they are not allocated.)
Full error checking has been added to these expressions. When
used without error checking, non-union build, use of these
expressions will incur no loss of efficiency. (In fact, XSET* is
now defined in terms of wrap_* in a non-union build.) In a union
build, you will also get no loss of efficiency provided that you
have a decent optimizing compiler, and a compiler that either
understands inlines or automatically inlines those particular
functions. (And since people don't normally do their production
builds on union, it doesn't matter.)
Update the sample Lisp object definition in lrecord.h accordingly.
dumper.c: Fix places in dumper that referenced wrap_object to reference
its new name, wrap_pointer_1.
buffer.c, bufslots.h, conslots.h, console.c, console.h, devslots.h, device.c, device.h, frame.c, frame.h, frameslots.h, window.c, window.h, winslots.h: -- Extract out the Lisp objects of `struct device' into devslots.h,
just like for the other structures.
-- Extract out the remaining (not copied into the window config)
Lisp objects in `struct window' into winslots.h; use different
macros (WINDOW_SLOT vs. WINDOW_SAVED_SLOT) to differentiate them.
-- Eliminate the `dead' flag of `struct frame', since it
duplicates information already available in `framemeths', and fix
FRAME_LIVE_P accordingly. (Devices and consoles already work this
way.)
-- In *slots.h, switch to system where MARKED_SLOT is automatically
undef'd at the end of the file. (Follows what winslots.h already
does.)
-- Update the comments at the beginning of *slots.h to be accurate.
-- When making any of the above objects dead, zero it out entirely
and reset all Lisp object slots to Qnil. (We were already doing
this somewhat, but not consistently.) This (1) Eliminates the
possibility of extra objects hanging around that ought to be
GC'd, (2) Causes an immediate crash if anyone tries to access a
structure in one of these objects, (3) Ensures consistent behavior
wrt dead objects.
dialog-msw.c: Use internal_object_printer, since this object should not escape.
---------------------------------------------------------------
FIXING A CRASH THAT I HIT ONCE (AND A RELATED BAD BEHAVIOR):
---------------------------------------------------------------
eval.c: Fix up some comments about the FSF implementation.
Fix two nasty bugs:
(1) condition_case_unwind frees the conses sitting in the
catch->tag slot too quickly, resulting in a crash that I hit.
(2) catches need to be unwound one at a time when calling
unwind-protect code, rather than all at once at the end; otherwise,
incorrect behavior can result. (A comment shows exactly how.)
backtrace.h: Improve comment about FSF differences in the handler stack.
---------------------------------------------------------------
FIXING A CRASH THAT I REPEATEDLY HIT WHEN USING THE MOUSE WHEEL
UNDER MSWINDOWS:
---------------------------------------------------------------
Basic idea: My crash is due either to a dead, non-marked,
GC-collected frame inside of a window mirror, or a prematurely
freed window mirror. We need to mark the Lisp objects inside of
window mirrors. Tracking the lifespan of window mirrors and
scrollbar instances is extremely hard, and there may well be
lurking bugs where such objects are freed too soon. The only safe
way to fix these problems (and it fixes both problems at once) is
to make both of these structures Lisp objects.
lrecord.h, emacs.c, inline.c, scrollbar-gtk.c, scrollbar-msw.c, scrollbar-x.c, scrollbar.c, scrollbar.h, symsinit.h: Make scrollbar instances actual Lisp objects. Mark the window
mirrors in them. inline.c needs to know about scrollbar.h now.
Record the new type in lrecord.h. Fix up scrollbar-*.c
appropriately. Create a hash table in scrollbar-msw.c so that the
scrollbar instances stored in scrollbar HWND's are properly
GC-protected. Create complex_vars_of_scrollbar_mswindows() to
create the hash table at startup, and call it from emacs.c. Don't
store the scrollbar instance as a property of the GTK scrollbar,
as it's not used and if we did this, we'd have to separately
GC-protect it in a hash table, like in MS Windows.
lrecord.h, frame.h, frame.c, frameslots.h, redisplay.c, window.c, window.h: Move mark_window_mirror from redisplay.c to window.c. Make window
mirrors actual Lisp objects. Tell lrecord.h about them. Change
the window mirror member of struct frame from a pointer to a Lisp
object, and add XWINDOW_MIRROR in appropriate places. Mark the
scrollbar instances in the window mirror.
redisplay.c, redisplay.h, alloc.c: Delete mark_redisplay. Don't call mark_redisplay. We now mark
frame-specific structures in mark_frame.
NOTE: I also deleted an extremely questionable call to
update_frame_window_mirrors(). It was extremely questionable
before, and now totally impossible, since it will create
Lisp objects during redisplay.
frame.c: Mark the scrollbar instances, which are now Lisp objects.
Call mark_gutter() here, not in mark_redisplay().
gutter.c: Update comments about correct marking.
---------------------------------------------------------------
ISSUES BROUGHT UP BY MARTIN:
---------------------------------------------------------------
buffer.h: Put back these macros the way Steve T and I think they ought to be.
I already explained in a previous changelog entry why I think these
macros should be the way I'd defined them. Once again:
We fix these macros so they don't care about the type of their
lvalues. The non-C-string equivalents of these already function
in the same way, and it's correct because it should be OK to pass
in a CBufbyte *, a BufByte *, a Char_Binary *, an UChar_Binary *,
etc. The whole reason for these different types is to work around
errors caused by signed-vs-unsigned non-matching types. Any
possible error that might be caught in a DFC macro would also be
caught wherever the argument is used elsewhere. So creating
multiple macro versions would add no useful error-checking and
just further complicate an already complicated area.
As for Martin's "ANSI aliasing" bug, XEmacs is not ANSI-aliasing
clean and probably never will be. Unless the board agrees to
change XEmacs in this way (and we really don't want to go down
that road), this is not a bug.
sound.h: Undo Martin's type change.
signal.c: Fix problem identified by Martin with Linux and g++ due to
non-standard declaration of setitimer().
systime.h: Update the docs for "qxe_" to point out why making the
encapsulation explicit is always the right way to go. (setitimer()
itself serves as an example.)
For 21.4:
update-elc-2.el: Correct misplaced parentheses, making lisp/mule not get
recompiled.
author | ben |
---|---|
date | Mon, 18 Jun 2001 07:10:32 +0000 |
parents | 38db05db9cb5 |
children | b39c14581166 |
line wrap: on
line source
/* Old synchronous subprocess invocation for XEmacs. Copyright (C) 1985, 86, 87, 88, 93, 94, 95 Free Software Foundation, Inc. This file is part of XEmacs. XEmacs is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. XEmacs is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with XEmacs; see the file COPYING. If not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ /* Synched up with: Mule 2.0, FSF 19.30. */ /* Partly sync'ed with 19.36.4 */ /* #### This ENTIRE file is only used in batch mode. We only need two things to get rid of both this and ntproc.c: -- my `stderr-proc' ws, which adds support for a separate stderr in asynch. subprocesses. (it's a feature in `old-call-process-internal'.) -- a noninteractive event loop that supports processes. */ #include <config.h> #include "lisp.h" #include "buffer.h" #include "commands.h" #include "insdel.h" #include "lstream.h" #include "process.h" #include "sysdep.h" #include "window.h" #ifdef FILE_CODING #include "file-coding.h" #endif #include "systime.h" #include "sysproc.h" #include "sysfile.h" /* Always include after sysproc.h */ #include "syssignal.h" /* Always include before systty.h */ #include "systty.h" #ifdef WIN32_NATIVE #define _P_NOWAIT 1 /* from process.h */ #include "nt.h" #endif #ifdef WIN32_NATIVE /* When we are starting external processes we need to know whether they take binary input (no conversion) or text input (\n is converted to \r\n). Similarly for output: if newlines are written as \r\n then it's text process output, otherwise it's binary. */ Lisp_Object Vbinary_process_input; Lisp_Object Vbinary_process_output; #endif /* WIN32_NATIVE */ Lisp_Object Vshell_file_name; /* The environment to pass to all subprocesses when they are started. This is in the semi-bogus format of ("VAR=VAL" "VAR2=VAL2" ... ) */ Lisp_Object Vprocess_environment; /* True iff we are about to fork off a synchronous process or if we are waiting for it. */ volatile int synch_process_alive; /* Nonzero => this is a string explaining death of synchronous subprocess. */ const char *synch_process_death; /* If synch_process_death is zero, this is exit code of synchronous subprocess. */ int synch_process_retcode; /* Clean up when exiting Fcall_process_internal. On Windows, delete the temporary file on any kind of termination. On Unix, kill the process and any children on termination by signal. */ /* Nonzero if this is termination due to exit. */ static int call_process_exited; Lisp_Object Vlisp_EXEC_SUFFIXES; static Lisp_Object call_process_kill (Lisp_Object fdpid) { Lisp_Object fd = Fcar (fdpid); Lisp_Object pid = Fcdr (fdpid); if (!NILP (fd)) close (XINT (fd)); if (!NILP (pid)) EMACS_KILLPG (XINT (pid), SIGKILL); synch_process_alive = 0; return Qnil; } static Lisp_Object call_process_cleanup (Lisp_Object fdpid) { int fd = XINT (Fcar (fdpid)); int pid = XINT (Fcdr (fdpid)); if (!call_process_exited && EMACS_KILLPG (pid, SIGINT) == 0) { int speccount = specpdl_depth (); record_unwind_protect (call_process_kill, fdpid); /* #### "c-G" -- need non-consing Single-key-description */ message ("Waiting for process to die...(type C-g again to kill it instantly)"); #ifdef WIN32_NATIVE { HANDLE pHandle = OpenProcess (PROCESS_ALL_ACCESS, 0, pid); if (pHandle == NULL) warn_when_safe (Qprocess, Qwarning, "cannot open process (PID %d) for cleanup", pid); else wait_for_termination (pHandle); } #else wait_for_termination (pid); #endif /* "Discard" the unwind protect. */ XCAR (fdpid) = Qnil; XCDR (fdpid) = Qnil; unbind_to (speccount, Qnil); message ("Waiting for process to die... done"); } synch_process_alive = 0; close (fd); return Qnil; } DEFUN ("old-call-process-internal", Fold_call_process_internal, 1, MANY, 0, /* Call PROGRAM synchronously in separate process, with coding-system specified. Arguments are (PROGRAM &optional INFILE BUFFER DISPLAY &rest ARGS). The program's input comes from file INFILE (nil means `/dev/null'). Insert output in BUFFER before point; t means current buffer; nil for BUFFER means discard it; 0 means discard and don't wait. BUFFER can also have the form (REAL-BUFFER STDERR-FILE); in that case, REAL-BUFFER says what to do with standard output, as above, while STDERR-FILE says what to do with standard error in the child. STDERR-FILE may be nil (discard standard error output), t (mix it with ordinary output), or a file name string. Fourth arg DISPLAY non-nil means redisplay buffer as output is inserted. Remaining arguments are strings passed as command arguments to PROGRAM. If BUFFER is 0, `call-process' returns immediately with value nil. Otherwise it waits for PROGRAM to terminate and returns a numeric exit status or a signal description string. If you quit, the process is killed with SIGINT, or SIGKILL if you quit again. */ (int nargs, Lisp_Object *args)) { /* This function can GC */ Lisp_Object infile, buffer, current_dir, display, path; int fd[2]; int filefd; #ifdef WIN32_NATIVE HANDLE pHandle; #endif int pid; char buf[16384]; char *bufptr = buf; int bufsize = 16384; int speccount = specpdl_depth (); struct gcpro gcpro1, gcpro2, gcpro3; char **new_argv = alloca_array (char *, max (2, nargs - 2)); /* File to use for stderr in the child. t means use same as standard output. */ Lisp_Object error_file; CHECK_STRING (args[0]); error_file = Qt; #if defined (NO_SUBPROCESSES) /* Without asynchronous processes we cannot have BUFFER == 0. */ if (nargs >= 3 && !INTP (args[2])) signal_error (Qunimplemented, "Operating system cannot handle asynchronous subprocesses", Qunbound); #endif /* NO_SUBPROCESSES */ /* Do all filename munging before building new_argv because GC in * Lisp code called by various filename-hacking routines might * relocate strings */ locate_file (Vexec_path, args[0], Vlisp_EXEC_SUFFIXES, &path, X_OK); /* Make sure that the child will be able to chdir to the current buffer's current directory, or its unhandled equivalent. [[ We can't just have the child check for an error when it does the chdir, since it's in a vfork. ]] -- not any more, we don't use vfork. -ben Note: These calls are spread out to insure that the return values of the calls (which may be newly-created strings) are properly GC-protected. */ { struct gcpro ngcpro1, ngcpro2; NGCPRO2 (current_dir, path); /* Caller gcprotects args[] */ current_dir = current_buffer->directory; /* If the current dir has no terminating slash, we'll get undesirable results, so put the slash back. */ current_dir = Ffile_name_as_directory (current_dir); current_dir = Funhandled_file_name_directory (current_dir); current_dir = expand_and_dir_to_file (current_dir, Qnil); #if 0 /* This is in FSF, but it breaks everything in the presence of ange-ftp-visited files, so away with it. */ if (NILP (Ffile_accessible_directory_p (current_dir))) signal_error (Qprocess_error, "Setting current directory", current_buffer->directory); #endif /* 0 */ NUNGCPRO; } GCPRO2 (current_dir, path); if (nargs >= 2 && ! NILP (args[1])) { struct gcpro ngcpro1; NGCPRO1 (current_buffer->directory); infile = Fexpand_file_name (args[1], current_buffer->directory); NUNGCPRO; CHECK_STRING (infile); } else infile = build_string (NULL_DEVICE); UNGCPRO; GCPRO3 (infile, current_dir, path); /* Fexpand_file_name might trash it */ if (nargs >= 3) { buffer = args[2]; /* If BUFFER is a list, its meaning is (BUFFER-FOR-STDOUT FILE-FOR-STDERR). */ if (CONSP (buffer)) { if (CONSP (XCDR (buffer))) { Lisp_Object file_for_stderr = XCAR (XCDR (buffer)); if (NILP (file_for_stderr) || EQ (Qt, file_for_stderr)) error_file = file_for_stderr; else error_file = Fexpand_file_name (file_for_stderr, Qnil); } buffer = XCAR (buffer); } if (!(EQ (buffer, Qnil) || EQ (buffer, Qt) || ZEROP (buffer))) { Lisp_Object spec_buffer = buffer; buffer = Fget_buffer (buffer); /* Mention the buffer name for a better error message. */ if (NILP (buffer)) CHECK_BUFFER (spec_buffer); CHECK_BUFFER (buffer); } } else buffer = Qnil; UNGCPRO; display = ((nargs >= 4) ? args[3] : Qnil); /* From here we assume we won't GC (unless an error is signaled). */ { REGISTER int i; for (i = 4; i < nargs; i++) { CHECK_STRING (args[i]); new_argv[i - 3] = (char *) XSTRING_DATA (args[i]); } } new_argv[max(nargs - 3,1)] = 0; if (NILP (path)) signal_error (Qprocess_error, "Searching for program", args[0]); new_argv[0] = (char *) XSTRING_DATA (path); filefd = open ((char *) XSTRING_DATA (infile), O_RDONLY | OPEN_BINARY, 0); if (filefd < 0) report_process_error ("Opening process input file", infile); if (INTP (buffer)) { fd[1] = open (NULL_DEVICE, O_WRONLY | OPEN_BINARY, 0); fd[0] = -1; } else { pipe (fd); #if 0 /* Replaced by close_process_descs */ set_exclusive_use (fd[0]); #endif } { /* child_setup must clobber environ in systems with true vfork. Protect it from permanent change. */ REGISTER char **save_environ = environ; REGISTER int fd1 = fd[1]; int fd_error = fd1; /* Record that we're about to create a synchronous process. */ synch_process_alive = 1; /* These vars record information from process termination. Clear them now before process can possibly terminate, to avoid timing error if process terminates soon. */ synch_process_death = 0; synch_process_retcode = 0; if (NILP (error_file)) fd_error = open (NULL_DEVICE, O_WRONLY | OPEN_BINARY); else if (STRINGP (error_file)) { fd_error = open ((const char *) XSTRING_DATA (error_file), #ifdef WIN32_NATIVE O_WRONLY | O_TRUNC | O_CREAT | O_TEXT, S_IREAD | S_IWRITE #else /* not WIN32_NATIVE */ O_WRONLY | O_TRUNC | O_CREAT | OPEN_BINARY, CREAT_MODE #endif /* not WIN32_NATIVE */ ); } if (fd_error < 0) { int save_errno = errno; close (filefd); close (fd[0]); if (fd1 >= 0) close (fd1); errno = save_errno; report_process_error ("Cannot open", error_file); } #ifdef WIN32_NATIVE pid = child_setup (filefd, fd1, fd_error, new_argv, (char *) XSTRING_DATA (current_dir)); if (!INTP (buffer)) { /* OpenProcess() as soon after child_setup as possible. It's too late once the process terminated. */ pHandle = OpenProcess(PROCESS_ALL_ACCESS, 0, pid); #if 0 if (pHandle == NULL) { /* #### seems to cause crash in unbind_to(...) below. APA */ warn_when_safe (Qprocess, Qwarning, "cannot open process to wait for"); } #endif } /* Close STDERR into the parent process. We no longer need it. */ if (fd_error >= 0) close (fd_error); #else /* not WIN32_NATIVE */ pid = fork (); if (pid == 0) { if (fd[0] >= 0) close (fd[0]); /* This is necessary because some shells may attempt to access the current controlling terminal and will hang if they are run in the background, as will be the case when XEmacs is started in the background. Martin Buchholz observed this problem running a subprocess that used zsh to call gzip to uncompress an info file. */ disconnect_controlling_terminal (); child_setup (filefd, fd1, fd_error, new_argv, (char *) XSTRING_DATA (current_dir)); } if (fd_error >= 0) close (fd_error); #endif /* not WIN32_NATIVE */ environ = save_environ; /* Close most of our fd's, but not fd[0] since we will use that to read input from. */ close (filefd); if (fd1 >= 0) close (fd1); } #ifndef WIN32_NATIVE if (pid < 0) { int save_errno = errno; if (fd[0] >= 0) close (fd[0]); errno = save_errno; report_process_error ("Doing fork", Qunbound); } #endif if (INTP (buffer)) { if (fd[0] >= 0) close (fd[0]); #if defined (NO_SUBPROCESSES) /* If Emacs has been built with asynchronous subprocess support, we don't need to do this, I think because it will then have the facilities for handling SIGCHLD. */ wait_without_blocking (); #endif /* NO_SUBPROCESSES */ return Qnil; } { int nread; int total_read = 0; Lisp_Object instream; struct gcpro ngcpro1; /* Enable sending signal if user quits below. */ call_process_exited = 0; record_unwind_protect (call_process_cleanup, Fcons (make_int (fd[0]), make_int (pid))); /* FSFmacs calls Fset_buffer() here. We don't have to because we can insert into buffers other than the current one. */ if (EQ (buffer, Qt)) XSETBUFFER (buffer, current_buffer); instream = make_filedesc_input_stream (fd[0], 0, -1, LSTR_ALLOW_QUIT); #ifdef FILE_CODING instream = make_decoding_input_stream (XLSTREAM (instream), Fget_coding_system (Vcoding_system_for_read)); Lstream_set_character_mode (XLSTREAM (instream)); #endif NGCPRO1 (instream); while (1) { QUIT; /* Repeatedly read until we've filled as much as possible of the buffer size we have. But don't read less than 1024--save that for the next bufferfull. */ nread = 0; while (nread < bufsize - 1024) { Lstream_data_count this_read = Lstream_read (XLSTREAM (instream), bufptr + nread, bufsize - nread); if (this_read < 0) goto give_up; if (this_read == 0) goto give_up_1; nread += this_read; } give_up_1: /* Now NREAD is the total amount of data in the buffer. */ if (nread == 0) break; #if 0 #ifdef WIN32_NATIVE /* Until we pull out of MULE things like make_decoding_input_stream(), we do the following which is less elegant. --marcpa */ /* We did. -- kkm */ { int lf_count = 0; if (NILP (Vbinary_process_output)) { nread = crlf_to_lf(nread, bufptr, &lf_count); } } #endif #endif total_read += nread; if (!NILP (buffer)) buffer_insert_raw_string (XBUFFER (buffer), (Bufbyte *) bufptr, nread); /* Make the buffer bigger as we continue to read more data, but not past 64k. */ if (bufsize < 64 * 1024 && total_read > 32 * bufsize) { bufsize *= 2; bufptr = (char *) alloca (bufsize); } if (!NILP (display) && INTERACTIVE) { redisplay (); } } give_up: Lstream_close (XLSTREAM (instream)); NUNGCPRO; QUIT; /* Wait for it to terminate, unless it already has. */ #ifdef WIN32_NATIVE wait_for_termination (pHandle); #else wait_for_termination (pid); #endif /* Don't kill any children that the subprocess may have left behind when exiting. */ call_process_exited = 1; unbind_to (speccount, Qnil); if (synch_process_death) return build_string (synch_process_death); return make_int (synch_process_retcode); } } /* Move the file descriptor FD so that its number is not less than MIN. * The original file descriptor remains open. */ static int relocate_fd (int fd, int min) { if (fd >= min) return fd; else { int newfd = dup (fd); if (newfd == -1) { stderr_out ("Error while setting up child: %s\n", strerror (errno)); _exit (1); } return relocate_fd (newfd, min); } } /* This is the last thing run in a newly forked inferior either synchronous or asynchronous. Copy descriptors IN, OUT and ERR as descriptors STDIN_FILENO, STDOUT_FILENO, and STDERR_FILENO. Initialize inferior's priority, pgrp, connected dir and environment. then exec another program based on new_argv. This function may change environ for the superior process. Therefore, the superior process must save and restore the value of environ around the fork and the call to this function. ENV is the environment for the subprocess. XEmacs: We've removed the SET_PGRP argument because it's already done by the callers of child_setup. CURRENT_DIR is an elisp string giving the path of the current directory the subprocess should have. Since we can't really signal a decent error from within the child, this should be verified as an executable directory by the parent. */ #ifdef WIN32_NATIVE int #else void #endif child_setup (int in, int out, int err, char **new_argv, const char *current_dir) { char **env; char *pwd; #ifdef WIN32_NATIVE int cpid; HANDLE handles[4]; #endif /* WIN32_NATIVE */ #ifdef SET_EMACS_PRIORITY if (emacs_priority != 0) nice (- emacs_priority); #endif /* Under Windows, we are not in a child process at all, so we should not close handles inherited from the parent -- we are the parent and doing so will screw up all manner of things! Similarly, most of the rest of the cleanup done in this function is not done under Windows. #### This entire child_setup() function is an utter and complete piece of shit. I would rewrite it, at the very least splitting out the Windows and non-Windows stuff into two completely different functions; but instead I'm trying to make it go away entirely, using the Lisp definition in process.el. What's left is to fix up the routines in event-msw.c (and in event-Xt.c and event-tty.c) to allow for stream devices to be handled correctly. There isn't much to do, in fact, and I'll fix it shortly. That way, the Lisp definition can be used non-interactively too. */ #if !defined (NO_SUBPROCESSES) && !defined (WIN32_NATIVE) /* Close Emacs's descriptors that this process should not have. */ close_process_descs (); #endif /* not NO_SUBPROCESSES */ #ifndef WIN32_NATIVE close_load_descs (); #endif /* Note that use of alloca is always safe here. It's obvious for systems that do not have true vfork or that have true (stack) alloca. If using vfork and C_ALLOCA it is safe because that changes the superior's static variables as if the superior had done alloca and will be cleaned up in the usual way. */ { REGISTER int i; i = strlen (current_dir); pwd = alloca_array (char, i + 6); memcpy (pwd, "PWD=", 4); memcpy (pwd + 4, current_dir, i); i += 4; if (!IS_DIRECTORY_SEP (pwd[i - 1])) pwd[i++] = DIRECTORY_SEP; pwd[i] = 0; /* We can't signal an Elisp error here; we're in a vfork. Since the callers check the current directory before forking, this should only return an error if the directory's permissions are changed between the check and this chdir, but we should at least check. */ if (chdir (pwd + 4) < 0) { /* Don't report the chdir error, or ange-ftp.el doesn't work. */ /* (FSFmacs does _exit (errno) here.) */ pwd = 0; } else { /* Strip trailing "/". Cretinous *[]&@$#^%@#$% Un*x */ /* leave "//" (from FSF) */ while (i > 6 && IS_DIRECTORY_SEP (pwd[i - 1])) pwd[--i] = 0; } } /* Set `env' to a vector of the strings in Vprocess_environment. */ /* + 2 to include PWD and terminating 0. */ env = alloca_array (char *, XINT (Flength (Vprocess_environment)) + 2); { REGISTER Lisp_Object tail; char **new_env = env; /* If we have a PWD envvar and we know the real current directory, pass one down, but with corrected value. */ if (pwd && getenv ("PWD")) *new_env++ = pwd; /* Copy the Vprocess_environment strings into new_env. */ for (tail = Vprocess_environment; CONSP (tail) && STRINGP (XCAR (tail)); tail = XCDR (tail)) { char **ep = env; char *envvar_external; TO_EXTERNAL_FORMAT (LISP_STRING, XCAR (tail), C_STRING_ALLOCA, envvar_external, Qfile_name); /* See if envvar_external duplicates any string already in the env. If so, don't put it in. When an env var has multiple definitions, we keep the definition that comes first in process-environment. */ for (; ep != new_env; ep++) { char *p = *ep, *q = envvar_external; while (1) { if (*q == 0) /* The string is malformed; might as well drop it. */ goto duplicate; if (*q != *p) break; if (*q == '=') goto duplicate; p++, q++; } } if (pwd && !strncmp ("PWD=", envvar_external, 4)) { *new_env++ = pwd; pwd = 0; } else *new_env++ = envvar_external; duplicate: ; } *new_env = 0; } #ifdef WIN32_NATIVE prepare_standard_handles (in, out, err, handles); set_process_dir (current_dir); #else /* not WIN32_NATIVE */ /* Make sure that in, out, and err are not actually already in descriptors zero, one, or two; this could happen if Emacs is started with its standard in, out, or error closed, as might happen under X. */ in = relocate_fd (in, 3); out = relocate_fd (out, 3); err = relocate_fd (err, 3); /* Set the standard input/output channels of the new process. */ close (STDIN_FILENO); close (STDOUT_FILENO); close (STDERR_FILENO); dup2 (in, STDIN_FILENO); dup2 (out, STDOUT_FILENO); dup2 (err, STDERR_FILENO); close (in); close (out); close (err); /* I can't think of any reason why child processes need any more than the standard 3 file descriptors. It would be cleaner to close just the ones that need to be, but the following brute force approach is certainly effective, and not too slow. */ { int fd; for (fd=3; fd<=64; fd++) close (fd); } #endif /* not WIN32_NATIVE */ #ifdef vipc something missing here; #endif /* vipc */ #ifdef WIN32_NATIVE /* Spawn the child. (See ntproc.c:Spawnve). */ cpid = spawnve (_P_NOWAIT, new_argv[0], (const char* const*)new_argv, (const char* const*)env); if (cpid == -1) /* An error occurred while trying to spawn the process. */ report_process_error ("Spawning child process", Qunbound); reset_standard_handles (in, out, err, handles); return cpid; #else /* not WIN32_NATIVE */ /* execvp does not accept an environment arg so the only way to pass this environment is to set environ. Our caller is responsible for restoring the ambient value of environ. */ environ = env; execvp (new_argv[0], new_argv); stdout_out ("Can't exec program %s\n", new_argv[0]); _exit (1); #endif /* not WIN32_NATIVE */ } static int getenv_internal (const Bufbyte *var, Bytecount varlen, Bufbyte **value, Bytecount *valuelen) { Lisp_Object scan; for (scan = Vprocess_environment; CONSP (scan); scan = XCDR (scan)) { Lisp_Object entry = XCAR (scan); if (STRINGP (entry) && XSTRING_LENGTH (entry) > varlen && XSTRING_BYTE (entry, varlen) == '=' #ifdef WIN32_NATIVE /* NT environment variables are case insensitive. */ && ! memicmp (XSTRING_DATA (entry), var, varlen) #else /* not WIN32_NATIVE */ && ! memcmp (XSTRING_DATA (entry), var, varlen) #endif /* not WIN32_NATIVE */ ) { *value = XSTRING_DATA (entry) + (varlen + 1); *valuelen = XSTRING_LENGTH (entry) - (varlen + 1); return 1; } } return 0; } DEFUN ("getenv", Fgetenv, 1, 2, "sEnvironment variable: \np", /* Return the value of environment variable VAR, as a string. VAR is a string, the name of the variable. When invoked interactively, prints the value in the echo area. */ (var, interactivep)) { Bufbyte *value; Bytecount valuelen; Lisp_Object v = Qnil; struct gcpro gcpro1; CHECK_STRING (var); GCPRO1 (v); if (getenv_internal (XSTRING_DATA (var), XSTRING_LENGTH (var), &value, &valuelen)) v = make_string (value, valuelen); if (!NILP (interactivep)) { if (NILP (v)) message ("%s not defined in environment", XSTRING_DATA (var)); else /* #### Should use Fprin1_to_string or Fprin1 to handle string containing quotes correctly. */ message ("\"%s\"", value); } RETURN_UNGCPRO (v); } /* A version of getenv that consults process_environment, easily callable from C. */ char * egetenv (const char *var) { /* This cannot GC -- 7-28-00 ben */ Bufbyte *value; Bytecount valuelen; if (getenv_internal ((const Bufbyte *) var, strlen (var), &value, &valuelen)) return (char *) value; else return 0; } void init_callproc (void) { /* This function can GC */ { /* jwz: always initialize Vprocess_environment, so that egetenv() works in temacs. */ char **envp; Vprocess_environment = Qnil; for (envp = environ; envp && *envp; envp++) Vprocess_environment = Fcons (build_ext_string (*envp, Qfile_name), Vprocess_environment); } { /* Initialize shell-file-name from environment variables or best guess. */ #ifdef WIN32_NATIVE const char *shell = egetenv ("SHELL"); if (!shell) shell = egetenv ("COMSPEC"); /* Should never happen! */ if (!shell) shell = (GetVersion () & 0x80000000 ? "command" : "cmd"); #else /* not WIN32_NATIVE */ const char *shell = egetenv ("SHELL"); if (!shell) shell = "/bin/sh"; #endif #if 0 /* defined (WIN32_NATIVE) */ /* BAD BAD BAD. We do not wanting to be passing an XEmacs-created SHELL var down to some inferior Cygwin process, which might get screwed up. There are a few broken apps (eterm/term.el, eterm/tshell.el, os-utils/terminal.el, texinfo/tex-mode.el) where this will cause problems. Those broken apps don't look at shell-file-name, instead just at explicit-shell-file-name, ESHELL and SHELL. They are apparently attempting to borrow what `M-x shell' uses, but that latter also looks at shell-file-name. What we want is for all of these apps to look at shell-file-name, so that the user can change the value of shell-file-name and everything will work out hunky-dorey. */ if (!egetenv ("SHELL")) { CBufbyte *faux_var = alloca_array (CBufbyte, 7 + strlen (shell)); sprintf (faux_var, "SHELL=%s", shell); Vprocess_environment = Fcons (build_string (faux_var), Vprocess_environment); } #endif /* 0 */ Vshell_file_name = build_string (shell); } } #if 0 void set_process_environment (void) { REGISTER char **envp; Vprocess_environment = Qnil; #ifndef CANNOT_DUMP if (initialized) #endif for (envp = environ; *envp; envp++) Vprocess_environment = Fcons (build_string (*envp), Vprocess_environment); } #endif /* unused */ void syms_of_callproc (void) { DEFSUBR (Fold_call_process_internal); DEFSUBR (Fgetenv); } void vars_of_callproc (void) { /* This function can GC */ #ifdef WIN32_NATIVE DEFVAR_LISP ("binary-process-input", &Vbinary_process_input /* *If non-nil then new subprocesses are assumed to take binary input. */ ); Vbinary_process_input = Qnil; DEFVAR_LISP ("binary-process-output", &Vbinary_process_output /* *If non-nil then new subprocesses are assumed to produce binary output. */ ); Vbinary_process_output = Qnil; #endif /* WIN32_NATIVE */ DEFVAR_LISP ("shell-file-name", &Vshell_file_name /* *File name to load inferior shells from. Initialized from the SHELL environment variable. */ ); DEFVAR_LISP ("process-environment", &Vprocess_environment /* List of environment variables for subprocesses to inherit. Each element should be a string of the form ENVVARNAME=VALUE. The environment which Emacs inherits is placed in this variable when Emacs starts. */ ); Vlisp_EXEC_SUFFIXES = build_string (EXEC_SUFFIXES); staticpro (&Vlisp_EXEC_SUFFIXES); }