274
|
1 /* Asynchronous subprocess implemenation for Win32
|
|
2 Copyright (C) 1985, 1986, 1987, 1988, 1992, 1993, 1994, 1995
|
|
3 Free Software Foundation, Inc.
|
|
4 Copyright (C) 1995 Sun Microsystems, Inc.
|
|
5 Copyright (C) 1995, 1996 Ben Wing.
|
|
6
|
|
7 This file is part of XEmacs.
|
|
8
|
|
9 XEmacs is free software; you can redistribute it and/or modify it
|
|
10 under the terms of the GNU General Public License as published by the
|
|
11 Free Software Foundation; either version 2, or (at your option) any
|
|
12 later version.
|
|
13
|
|
14 XEmacs is distributed in the hope that it will be useful, but WITHOUT
|
|
15 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
16 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
17 for more details.
|
|
18
|
|
19 You should have received a copy of the GNU General Public License
|
|
20 along with XEmacs; see the file COPYING. If not, write to
|
|
21 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
22 Boston, MA 02111-1307, USA. */
|
|
23
|
|
24 /* Written by Kirill M. Katsnelson <kkm@kis.ru>, April 1998 */
|
|
25
|
|
26 #include <config.h>
|
|
27 #include "lisp.h"
|
|
28
|
|
29 #include "hash.h"
|
|
30 #include "lstream.h"
|
|
31 #include "process.h"
|
|
32 #include "procimpl.h"
|
|
33
|
|
34 #include <windows.h>
|
|
35
|
|
36 /* Implemenation-specific data. Pointed to by Lisp_Process->process_data */
|
|
37 struct nt_process_data
|
|
38 {
|
|
39 HANDLE h_process;
|
|
40 };
|
|
41
|
|
42 #define NT_DATA(p) ((struct nt_process_data*)((p)->process_data))
|
|
43
|
|
44 /*-----------------------------------------------------------------------*/
|
|
45 /* Process helpers */
|
|
46 /*-----------------------------------------------------------------------*/
|
|
47
|
|
48 /* #### Ok, a pretty frame is here, and a nifty text is in. Now,
|
|
49 any function around here to be put here? Yahoooo.... */
|
|
50
|
|
51 /*-----------------------------------------------------------------------*/
|
|
52 /* Process methods */
|
|
53 /*-----------------------------------------------------------------------*/
|
|
54
|
|
55 /*
|
|
56 * Allocate and initialize Lisp_Process->process_data
|
|
57 */
|
|
58
|
|
59 static void
|
|
60 nt_alloc_process_data (struct Lisp_Process *p)
|
|
61 {
|
|
62 p->process_data = xnew (struct nt_process_data);
|
|
63 }
|
|
64
|
|
65 #if 0 /* #### Need this method? */
|
|
66 /*
|
|
67 * Mark any Lisp objects in Lisp_Process->process_data
|
|
68 */
|
|
69
|
|
70 static void
|
|
71 nt_mark_process_data (struct Lisp_Process *proc,
|
|
72 void (*markobj) (Lisp_Object))
|
|
73 {
|
|
74 }
|
|
75 #endif
|
|
76
|
|
77 static void
|
|
78 nt_finalize_process_data (struct Lisp_Process *p, int for_disksave)
|
|
79 {
|
|
80 assert (!for_disksave);
|
|
81 if (NT_DATA(p)->h_process)
|
|
82 CloseHandle (NT_DATA(p)->h_process);
|
|
83 }
|
|
84
|
|
85 #if 0 /* #### Need this method? */
|
|
86 /*
|
|
87 * Initialize XEmacs process implemenation once
|
|
88 */
|
|
89
|
|
90 static void
|
|
91 nt_init_process (void)
|
|
92 {
|
|
93 }
|
|
94 #endif
|
|
95
|
|
96 #if 0 /* #### Need this method? */
|
|
97 /*
|
|
98 * Initialize any process local data. This is called when newly
|
|
99 * created process is connected to real OS file handles. The
|
|
100 * handles are generally represented by void* type, but are
|
|
101 * of type HANDLE for Win32
|
|
102 */
|
|
103
|
|
104 static void
|
|
105 nt_init_process_io_handles (struct Lisp_Process *p, void* in, void* out, int flags)
|
|
106 {
|
|
107 }
|
|
108 #endif
|
|
109
|
|
110 /*
|
|
111 * Fork off a subprocess. P is a pointer to newly created subprocess
|
|
112 * object. If this function signals, the caller is responsible for
|
|
113 * deleting (and finalizing) the process object.
|
|
114 *
|
|
115 * The method must return PID of the new proces, a (positive??? ####) number
|
|
116 * which fits into Lisp_Int. No return value indicates an error, the method
|
|
117 * must signal an error instead.
|
|
118 */
|
|
119
|
|
120 /* #### This function completely ignores Vprocess_environment */
|
|
121
|
|
122 static int
|
|
123 nt_create_process (struct Lisp_Process *p,
|
|
124 char **argv, CONST char *current_dir)
|
|
125 {
|
|
126 HANDLE hmyshove, hmyslurp, hprocin, hprocout;
|
|
127 LPTSTR command_line;
|
|
128
|
|
129 /* Create two unidirectional named pipes */
|
|
130 {
|
|
131 HANDLE htmp;
|
|
132 SECURITY_ATTRIBUTES sa;
|
|
133
|
|
134 sa.nLength = sizeof(sa);
|
|
135 sa.bInheritHandle = TRUE;
|
|
136 sa.lpSecurityDescriptor = NULL;
|
|
137
|
|
138 CreatePipe (&hprocin, &hmyshove, &sa, 0);
|
|
139 CreatePipe (&hmyslurp, &hprocout, &sa, 0);
|
|
140
|
|
141 /* Stupid Win32 allows to create a pipe with *both* ends either
|
|
142 inheritable or not. We need process ends inheritable, and local
|
|
143 ends not inheritable. */
|
|
144 /* #### Perhaps even stupider me does not know how to do this better */
|
|
145 DuplicateHandle (GetCurrentProcess(), hmyshove, GetCurrentProcess(), &htmp,
|
|
146 0, FALSE, DUPLICATE_CLOSE_SOURCE | DUPLICATE_SAME_ACCESS);
|
|
147 hmyshove = htmp;
|
|
148 DuplicateHandle (GetCurrentProcess(), hmyslurp, GetCurrentProcess(), &htmp,
|
|
149 0, FALSE, DUPLICATE_CLOSE_SOURCE | DUPLICATE_SAME_ACCESS);
|
|
150 hmyslurp = htmp;
|
|
151 }
|
|
152
|
|
153 /* Convert an argv vector into Win32 style command line.
|
|
154
|
|
155 #### This works only for cmd, and not for cygwin bash. Perhaps,
|
|
156 instead of ad-hoc fiddling with different methods for quoting
|
|
157 process arguments in ntproc.c (disgust shudder), this must call a
|
|
158 smart lisp routine. The code here will be a fallback, if the
|
|
159 lisp function is not specified.
|
|
160 */
|
|
161 {
|
|
162 char** thisarg;
|
|
163 size_t size = 1;
|
|
164
|
|
165 for (thisarg = argv; *thisarg; ++thisarg)
|
|
166 size += strlen (*thisarg) + 1;
|
|
167
|
|
168 command_line = alloca_array (char, size);
|
|
169 *command_line = 0;
|
|
170
|
|
171 for (thisarg = argv; *thisarg; ++thisarg)
|
|
172 {
|
|
173 if (thisarg != argv)
|
|
174 strcat (command_line, " ");
|
|
175 strcat (command_line, *thisarg);
|
|
176 }
|
|
177 }
|
|
178
|
|
179 /* Create process */
|
|
180 {
|
|
181 STARTUPINFO si;
|
|
182 PROCESS_INFORMATION pi;
|
|
183 DWORD err;
|
|
184 BOOL windowed;
|
|
185
|
|
186 xzero (si);
|
|
187 si.hStdInput = hprocin;
|
|
188 si.hStdOutput = hprocout;
|
|
189 si.hStdError = hprocout;
|
|
190 si.wShowWindow = SW_HIDE;
|
|
191 si.dwFlags = STARTF_USESTDHANDLES | STARTF_USESHOWWINDOW;
|
|
192
|
|
193 err = (CreateProcess (NULL, command_line, NULL, NULL, TRUE,
|
|
194 CREATE_NEW_CONSOLE | CREATE_NEW_PROCESS_GROUP | CREATE_SUSPENDED,
|
|
195 NULL, current_dir, &si, &pi)
|
|
196 ? 0 : GetLastError ());
|
|
197
|
|
198 CloseHandle (hprocin);
|
|
199 CloseHandle (hprocout);
|
|
200
|
|
201 /* See if we succeeded with process creation */
|
|
202 if (err)
|
|
203 {
|
|
204 process_error__One_of_those_nasty_uses_for_goto_statement:
|
|
205 CloseHandle (hmyshove);
|
|
206 CloseHandle (hmyslurp);
|
|
207 error ("Cannot start \"%s\": error code was %lu", argv[0], err);
|
|
208 }
|
|
209
|
|
210 /* Determine if the new process is a windowed one */
|
|
211 windowed = WaitForInputIdle (pi.hProcess, 100) == WAIT_TIMEOUT;
|
|
212 if (windowed)
|
|
213 {
|
|
214 /* We restart windowed process fire-and forget style, and
|
|
215 indicate successful process creation, just as if the
|
|
216 process ended instantly upon launching */
|
|
217 CloseHandle (hmyshove);
|
|
218 CloseHandle (hmyslurp);
|
|
219 /* TerminateProcess is safe becuase the process is not yet
|
|
220 running */
|
|
221 TerminateProcess (pi.hProcess, 0);
|
|
222 si.dwFlags = STARTF_USESHOWWINDOW;
|
|
223 si.wShowWindow = SW_SHOWNORMAL;
|
|
224 if (!CreateProcess (NULL, command_line, NULL, NULL, FALSE,
|
|
225 DETACHED_PROCESS , NULL, current_dir, &si, &pi))
|
|
226 {
|
|
227 err = GetLastError ();
|
|
228 goto process_error__One_of_those_nasty_uses_for_goto_statement;
|
|
229 }
|
|
230
|
|
231 /* We just launched a windowed process. Fake it as if a
|
|
232 process launched has already ended */
|
|
233 p->status_symbol = Qexit;
|
|
234
|
|
235 /* Get rid of process and thread handles */
|
|
236 CloseHandle (pi.hThread);
|
|
237 CloseHandle (pi.hProcess);
|
|
238 }
|
|
239 else
|
|
240 {
|
|
241 /* Just started a console subprocess */
|
|
242
|
|
243 NT_DATA(p)->h_process = pi.hProcess;
|
|
244
|
|
245 init_process_io_handles (p, (void*)hmyslurp, (void*)hmyshove, 0);
|
|
246
|
|
247 /* We created it suspended. Resume the only thread */
|
|
248 ResumeThread (pi.hThread);
|
|
249 CloseHandle (pi.hThread);
|
|
250 }
|
|
251
|
|
252 return ((int)pi.dwProcessId < 0
|
|
253 ? -(int)pi.dwProcessId : (int)pi.dwProcessId);
|
|
254 }
|
|
255 }
|
|
256
|
|
257 /*
|
|
258 * This method is called to update status fields of the process
|
|
259 * structure. If the process has not existed, this method is expected
|
|
260 * to do nothing.
|
|
261 *
|
|
262 * The method is called only for real child processes.
|
|
263 */
|
|
264
|
|
265 static void
|
|
266 nt_update_status_if_terminated (struct Lisp_Process* p)
|
|
267 {
|
|
268 DWORD exit_code;
|
|
269 if (GetExitCodeProcess (NT_DATA(p)->h_process, &exit_code)
|
|
270 && exit_code != STILL_ACTIVE)
|
|
271 {
|
|
272 p->tick++;
|
|
273 p->core_dumped = 0;
|
|
274 /* The exit code can be a code returned by process, or an
|
|
275 NTSTATUS value. We cannot accurately handle the latter since
|
|
276 it is a full 32 bit integer */
|
|
277 if (exit_code & 0xC0000000)
|
|
278 {
|
|
279 p->status_symbol = Qsignal;
|
|
280 p->exit_code = exit_code & 0x1FFFFFFF;
|
|
281 }
|
|
282 else
|
|
283 {
|
|
284 p->status_symbol = Qexit;
|
|
285 p->exit_code = exit_code;
|
|
286 }
|
|
287 }
|
|
288 }
|
|
289
|
|
290 /*
|
|
291 * Stuff the entire contents of LSTREAM to the process ouptut pipe
|
|
292 */
|
|
293
|
|
294 /* #### If only this function could be somehow merged with
|
|
295 unix_send_process... */
|
|
296
|
|
297 static void
|
|
298 nt_send_process (Lisp_Object proc, struct lstream* lstream)
|
|
299 {
|
|
300 struct Lisp_Process *p = XPROCESS (proc);
|
|
301
|
|
302 /* use a reasonable-sized buffer (somewhere around the size of the
|
|
303 stream buffer) so as to avoid inundating the stream with blocked
|
|
304 data. */
|
|
305 Bufbyte chunkbuf[512];
|
|
306 Bytecount chunklen;
|
|
307
|
|
308 while (1)
|
|
309 {
|
|
310 int writeret;
|
|
311
|
|
312 chunklen = Lstream_read (lstream, chunkbuf, 512);
|
|
313 if (chunklen <= 0)
|
|
314 break; /* perhaps should abort() if < 0?
|
|
315 This should never happen. */
|
|
316
|
|
317 /* Lstream_write() will never successfully write less than the
|
|
318 amount sent in. In the worst case, it just buffers the
|
|
319 unwritten data. */
|
|
320 writeret = Lstream_write (XLSTREAM (DATA_OUTSTREAM(p)), chunkbuf,
|
|
321 chunklen);
|
|
322 if (writeret < 0)
|
|
323 {
|
|
324 p->status_symbol = Qexit;
|
|
325 p->exit_code = ERROR_BROKEN_PIPE;
|
|
326 p->core_dumped = 0;
|
|
327 p->tick++;
|
|
328 process_tick++;
|
|
329 deactivate_process (proc);
|
|
330 error ("Broken pipe error sending to process %s; closed it",
|
|
331 XSTRING_DATA (p->name));
|
|
332 }
|
|
333
|
|
334 while (Lstream_was_blocked_p (XLSTREAM (p->pipe_outstream)))
|
|
335 {
|
|
336 /* Buffer is full. Wait, accepting input; that may allow
|
|
337 the program to finish doing output and read more. */
|
|
338 Faccept_process_output (Qnil, make_int (1), Qnil);
|
|
339 Lstream_flush (XLSTREAM (p->pipe_outstream));
|
|
340 }
|
|
341 }
|
|
342 Lstream_flush (XLSTREAM (DATA_OUTSTREAM(p)));
|
|
343 }
|
|
344
|
|
345 /*-----------------------------------------------------------------------*/
|
|
346 /* Initialization */
|
|
347 /*-----------------------------------------------------------------------*/
|
|
348
|
|
349 void
|
|
350 process_type_create_nt (void)
|
|
351 {
|
|
352 PROCESS_HAS_METHOD (nt, alloc_process_data);
|
|
353 PROCESS_HAS_METHOD (nt, finalize_process_data);
|
|
354 /* PROCESS_HAS_METHOD (nt, mark_process_data); */
|
|
355 /* PROCESS_HAS_METHOD (nt, init_process); */
|
|
356 /* PROCESS_HAS_METHOD (nt, init_process_io_handles); */
|
|
357 PROCESS_HAS_METHOD (nt, create_process);
|
|
358 PROCESS_HAS_METHOD (nt, update_status_if_terminated);
|
|
359 PROCESS_HAS_METHOD (nt, send_process);
|
|
360 /* PROCESS_HAS_METHOD (nt, kill_child_process); */
|
|
361 /* PROCESS_HAS_METHOD (nt, kill_process_by_pid); */
|
|
362 #if 0 /* Yet todo */
|
|
363 #ifdef HAVE_SOCKETS
|
|
364 PROCESS_HAS_METHOD (nt, canonicalize_host_name);
|
|
365 PROCESS_HAS_METHOD (nt, open_network_stream);
|
|
366 #ifdef HAVE_MULTICAST
|
|
367 PROCESS_HAS_METHOD (nt, open_multicast_group);
|
|
368 #endif
|
|
369 #endif
|
|
370 #endif
|
|
371 }
|
|
372
|
|
373 void
|
|
374 vars_of_process_nt (void)
|
|
375 {
|
|
376 }
|
|
377
|