]> git.saurik.com Git - wxWidgets.git/blame - src/os2/utilsexc.cpp
Restore code for closing inherited file descriptors in the child.
[wxWidgets.git] / src / os2 / utilsexc.cpp
CommitLineData
0e320a79 1/////////////////////////////////////////////////////////////////////////////
521bf4ff 2// Name: src/os2/utilsexec.cpp
d90895ac
DW
3// Purpose: Various utilities
4// Author: David Webster
0e320a79 5// Modified by:
d90895ac 6// Created: 10/17/99
0e320a79 7// RCS-ID: $Id$
d90895ac 8// Copyright: (c) David Webster
65571936 9// Licence: wxWindows licence
0e320a79
DW
10/////////////////////////////////////////////////////////////////////////////
11
d90895ac
DW
12// For compilers that support precompilation, includes "wx.h".
13#include "wx/wxprec.h"
0e320a79 14
d90895ac 15#ifndef WX_PRECOMP
521bf4ff
WS
16 #include "wx/utils.h"
17 #include "wx/app.h"
18 #include "wx/intl.h"
e4db172a 19 #include "wx/log.h"
d90895ac
DW
20#endif
21
d90895ac
DW
22#include "wx/process.h"
23
24#include "wx/os2/private.h"
25
f38374d0 26#define PURE_32
9aab8b20 27#ifndef __EMX__
6670f564
WS
28 #include <upm.h>
29 #ifndef __WATCOMC__
30 #include <netcons.h>
31 #include <netbios.h>
32 #endif
9aab8b20 33#endif
c5fb56c0 34
d90895ac 35#include <ctype.h>
9aab8b20
SN
36#ifdef __EMX__
37#include <dirent.h>
9aab8b20 38#endif
d90895ac
DW
39
40#include <sys/stat.h>
41#include <io.h>
0e320a79
DW
42
43#include <stdio.h>
44#include <stdlib.h>
45#include <string.h>
d90895ac
DW
46#include <errno.h>
47#include <stdarg.h>
48
9aab8b20 49
d90895ac
DW
50// this message is sent when the process we're waiting for terminates
51#define wxWM_PROC_TERMINATED (WM_USER + 10000)
52
dde11e60
DW
53#ifndef NO_ERROR
54# define NO_ERROR 0
55#endif
56
d90895ac
DW
57// structure describing the process we're being waiting for
58struct wxExecuteData
59{
60public:
61 ~wxExecuteData()
62 {
9aab8b20 63// cout << "Closing thread: " << endl;
dde11e60 64 DosExit(EXIT_PROCESS, 0);
d90895ac
DW
65 }
66
6670f564
WS
67 HWND hWnd; // window to send wxWM_PROC_TERMINATED to [not used]
68 RESULTCODES vResultCodes;
69 wxProcess* pHandler;
70 ULONG ulExitCode; // the exit code of the process
71 bool bState; // set to false when the process finishes
d90895ac
DW
72};
73
6670f564 74static ULONG wxExecuteThread(wxExecuteData* pData)
d90895ac 75{
6670f564
WS
76 ULONG ulRc;
77 PID vPidChild;
dde11e60 78
9aab8b20 79// cout << "Executing thread: " << endl;
c5fb56c0
DW
80
81 ulRc = ::DosWaitChild( DCWA_PROCESSTREE
82 ,DCWW_NOWAIT
dde11e60
DW
83 ,&pData->vResultCodes
84 ,&vPidChild
85 ,pData->vResultCodes.codeTerminate // process PID to look at
86 );
87 if (ulRc != NO_ERROR)
d90895ac 88 {
2173b18f 89 wxLogLastError(wxT("DosWaitChild"));
d90895ac 90 }
dde11e60 91 delete pData;
d90895ac
DW
92 return 0;
93}
94
c5fb56c0
DW
95// window procedure of a hidden window which is created just to receive
96// the notification message when a process exits
6670f564
WS
97MRESULT APIENTRY wxExecuteWindowCbk( HWND hWnd,
98 ULONG ulMessage,
99 MPARAM WXUNUSED(wParam),
100 MPARAM lParam)
d90895ac 101{
dde11e60 102 if (ulMessage == wxWM_PROC_TERMINATED)
d90895ac 103 {
dde11e60 104 wxExecuteData* pData = (wxExecuteData *)lParam;
d90895ac 105
dde11e60 106 if (pData->pHandler)
d90895ac 107 {
dde11e60
DW
108 pData->pHandler->OnTerminate( (int)pData->vResultCodes.codeTerminate
109 ,(int)pData->vResultCodes.codeResult
110 );
d90895ac
DW
111 }
112
dde11e60 113 if (pData->bState)
d90895ac
DW
114 {
115 // we're executing synchronously, tell the waiting thread
116 // that the process finished
dde11e60 117 pData->bState = 0;
d90895ac
DW
118 }
119 else
120 {
121 // asynchronous execution - we should do the clean up
dde11e60 122 delete pData;
d90895ac 123 }
dde11e60 124 ::WinDestroyWindow(hWnd); // we don't need it any more
d90895ac 125 }
d90895ac
DW
126 return 0;
127}
128
6670f564
WS
129long wxExecute( const wxString& rCommand,
130 int flags,
164db92c
VZ
131 wxProcess* pHandler,
132 const wxExecuteEnv *env)
0e320a79 133{
6670f564 134 if (rCommand.empty())
c5fb56c0 135 {
9aab8b20 136// cout << "empty command in wxExecute." << endl;
c5fb56c0
DW
137 return 0;
138 }
d90895ac
DW
139
140 // create the process
dde11e60
DW
141 UCHAR vLoadError[CCHMAXPATH] = {0};
142 RESULTCODES vResultCodes = {0};
143 ULONG ulExecFlag;
144 PSZ zArgs = NULL;
145 PSZ zEnvs = NULL;
dde11e60 146 APIRET rc;
dde11e60
DW
147 TID vTID;
148
171d29f9 149 if (flags & wxEXEC_SYNC)
dde11e60
DW
150 ulExecFlag = EXEC_SYNC;
151 else
152 ulExecFlag = EXEC_ASYNCRESULT;
153
c5fb56c0
DW
154 rc = ::DosExecPgm( (PCHAR)vLoadError
155 ,sizeof(vLoadError)
156 ,ulExecFlag
157 ,zArgs
158 ,zEnvs
159 ,&vResultCodes
08bee49e 160 ,rCommand.c_str()
c5fb56c0
DW
161 );
162 if (rc != NO_ERROR)
d90895ac 163 {
c5fb56c0 164 wxLogSysError(_("Execution of command '%s' failed with error: %ul"), rCommand.c_str(), rc);
d90895ac
DW
165 return 0;
166 }
9aab8b20 167// cout << "Executing: " << rCommand.c_str() << endl;
d90895ac 168 // Alloc data
dde11e60 169 wxExecuteData* pData = new wxExecuteData;
d90895ac 170
dde11e60
DW
171 pData->vResultCodes = vResultCodes;
172 pData->hWnd = NULLHANDLE;
171d29f9
VZ
173 pData->bState = (flags & wxEXEC_SYNC) != 0;
174 if (flags & wxEXEC_SYNC)
dde11e60
DW
175 {
176 wxASSERT_MSG(!pHandler, wxT("wxProcess param ignored for sync execution"));
177 pData->pHandler = NULL;
d90895ac
DW
178 }
179 else
180 {
181 // may be NULL or not
dde11e60 182 pData->pHandler = pHandler;
d90895ac
DW
183 }
184
dde11e60
DW
185 rc = ::DosCreateThread( &vTID
186 ,(PFNTHREAD)&wxExecuteThread
187 ,(ULONG)pData
188 ,CREATE_READY|STACK_SPARSE
189 ,8192
190 );
191 if (rc != NO_ERROR)
d90895ac 192 {
2173b18f 193 wxLogLastError(wxT("CreateThread in wxExecute"));
dde11e60 194 delete pData;
d90895ac
DW
195
196 // the process still started up successfully...
dde11e60 197 return vResultCodes.codeTerminate;
d90895ac 198 }
171d29f9 199 if (!(flags & wxEXEC_SYNC))
d90895ac 200 {
d90895ac 201 // return the pid
c5fb56c0
DW
202 // warning: don't exit your app unless you actively
203 // kill and cleanup you child processes
204 // Maybe detach the process here???
205 // If cmd.exe need to pass DETACH to detach the process here
dde11e60 206 return vResultCodes.codeTerminate;
d90895ac 207 }
c5fb56c0
DW
208
209 // waiting until command executed
dde11e60 210 ::DosWaitThread(&vTID, DCWW_WAIT);
d90895ac 211
dde11e60
DW
212 ULONG ulExitCode = pData->vResultCodes.codeResult;
213 delete pData;
d90895ac
DW
214
215 // return the exit code
dde11e60 216 return (long)ulExitCode;
0e320a79 217}
d90895ac 218
dde11e60
DW
219long wxExecute(
220 char** ppArgv
171d29f9 221, int flags
dde11e60 222, wxProcess* pHandler
164db92c 223, const wxExecuteEnv *env
dde11e60 224)
d90895ac 225{
dde11e60 226 wxString sCommand;
d90895ac 227
dde11e60 228 while (*ppArgv != NULL)
d90895ac 229 {
0fba44b4
DW
230 wxString sArg((wxChar*)(*ppArgv++));
231
232
233 sCommand << sArg.c_str() << ' ';
d90895ac 234 }
dde11e60
DW
235 sCommand.RemoveLast();
236 return wxExecute( sCommand
171d29f9 237 ,flags
dde11e60 238 ,pHandler
164db92c 239 , env
dde11e60 240 );
d90895ac
DW
241}
242
63a3cd7a 243bool wxGetFullHostName( wxChar* zBuf, int nMaxSize)
d90895ac 244{
63a3cd7a 245 return wxGetHostName( zBuf, nMaxSize );
d90895ac 246}