]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/unix/utilsunx.cpp
Corrected byte swapping macros.
[wxWidgets.git] / src / unix / utilsunx.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: utilsunx.cpp
3// Purpose: generic Unix implementation of many wx functions
4// Author: Vadim Zeitlin
5// Id: $Id$
6// Copyright: (c) 1998 Robert Roebling, Vadim Zeitlin
7// Licence: wxWindows licence
8/////////////////////////////////////////////////////////////////////////////
9
10// ============================================================================
11// declarations
12// ============================================================================
13
14// ----------------------------------------------------------------------------
15// headers
16// ----------------------------------------------------------------------------
17
18#include "wx/defs.h"
19#include "wx/string.h"
20
21#include "wx/intl.h"
22#include "wx/log.h"
23
24#include "wx/utils.h"
25#include "wx/process.h"
26
27#include "wx/unix/execute.h"
28
29#include <stdarg.h>
30#include <dirent.h>
31#include <string.h>
32#include <sys/stat.h>
33#include <sys/types.h>
34#include <unistd.h>
35#include <sys/wait.h>
36#include <pwd.h>
37#include <errno.h>
38#include <netdb.h>
39#include <signal.h>
40#include <fcntl.h> // for O_WRONLY and friends
41#include <time.h> // nanosleep() and/or usleep()
42#include <ctype.h> // isspace()
43
44// JACS: needed for FD_SETSIZE
45#include <sys/time.h>
46
47#if HAVE_UNAME
48 #include <sys/utsname.h> // for uname()
49#endif // HAVE_UNAME
50
51// ----------------------------------------------------------------------------
52// conditional compilation
53// ----------------------------------------------------------------------------
54
55// many versions of Unices have this function, but it is not defined in system
56// headers - please add your system here if it is the case for your OS.
57// SunOS < 5.6 (i.e. Solaris < 2.6) and DG-UX are like this.
58#if !defined(HAVE_USLEEP) && \
59 (defined(__SUN__) && !defined(__SunOs_5_6) && \
60 !defined(__SunOs_5_7) && !defined(__SUNPRO_CC)) || \
61 defined(__osf__)
62 extern "C"
63 {
64 #ifdef __SUN__
65 int usleep(unsigned int usec);
66 #else // !Sun
67 void usleep(unsigned long usec);
68 #endif // Sun/!Sun
69 };
70#endif // Unices without usleep()
71
72// ============================================================================
73// implementation
74// ============================================================================
75
76// ----------------------------------------------------------------------------
77// sleeping
78// ----------------------------------------------------------------------------
79
80void wxSleep(int nSecs)
81{
82 sleep(nSecs);
83}
84
85void wxUsleep(unsigned long milliseconds)
86{
87#if HAVE_NANOSLEEP
88 timespec tmReq;
89 tmReq.tv_sec = milliseconds / 1000;
90 tmReq.tv_nsec = (milliseconds % 1000) * 1000 * 1000;
91
92 // we're not interested in remaining time nor in return value
93 (void)nanosleep(&tmReq, (timespec *)NULL);
94#elif HAVE_USLEEP
95 // uncomment this if you feel brave or if you are sure that your version
96 // of Solaris has a safe usleep() function but please notice that usleep()
97 // is known to lead to crashes in MT programs in Solaris 2.[67] and is not
98 // documented as MT-Safe
99 #if defined(__SUN__) && wxUSE_THREADS
100 #error "usleep() cannot be used in MT programs under Solaris."
101 #endif // Sun
102
103 usleep(milliseconds * 1000); // usleep(3) wants microseconds
104#else // !sleep function
105 #error "usleep() or nanosleep() function required for wxUsleep"
106#endif // sleep function
107}
108
109// ----------------------------------------------------------------------------
110// process management
111// ----------------------------------------------------------------------------
112
113int wxKill(long pid, wxSignal sig)
114{
115 return kill(pid, (int)sig);
116}
117
118#define WXEXECUTE_NARGS 127
119
120long wxExecute( const wxString& command, bool sync, wxProcess *process )
121{
122 wxCHECK_MSG( !command.IsEmpty(), 0, _T("can't exec empty command") );
123
124 int argc = 0;
125 wxChar *argv[WXEXECUTE_NARGS];
126 wxString argument;
127 const wxChar *cptr = command.c_str();
128 wxChar quotechar = _T('\0'); // is arg quoted?
129 bool escaped = FALSE;
130
131 // split the command line in arguments
132 do
133 {
134 argument=_T("");
135 quotechar = _T('\0');
136
137 // eat leading whitespace:
138 while ( wxIsspace(*cptr) )
139 cptr++;
140
141 if ( *cptr == _T('\'') || *cptr == _T('"') )
142 quotechar = *cptr++;
143
144 do
145 {
146 if ( *cptr == _T('\\') && ! escaped )
147 {
148 escaped = TRUE;
149 cptr++;
150 continue;
151 }
152
153 // all other characters:
154 argument += *cptr++;
155 escaped = FALSE;
156
157 // have we reached the end of the argument?
158 if ( (*cptr == quotechar && ! escaped)
159 || (quotechar == _T('\0') && wxIsspace(*cptr))
160 || *cptr == _T('\0') )
161 {
162 wxASSERT_MSG( argc < WXEXECUTE_NARGS,
163 _T("too many arguments in wxExecute") );
164
165 argv[argc] = new wxChar[argument.length() + 1];
166 wxStrcpy(argv[argc], argument.c_str());
167 argc++;
168
169 // if not at end of buffer, swallow last character:
170 if(*cptr)
171 cptr++;
172
173 break; // done with this one, start over
174 }
175 } while(*cptr);
176 } while(*cptr);
177 argv[argc] = NULL;
178
179 // do execute the command
180 long lRc = wxExecute(argv, sync, process);
181
182 // clean up
183 argc = 0;
184 while( argv[argc] )
185 delete [] argv[argc++];
186
187 return lRc;
188}
189
190bool wxShell(const wxString& command)
191{
192 wxString cmd;
193 if ( !!command )
194 cmd.Printf(_T("xterm -e %s"), command.c_str());
195 else
196 cmd = command;
197
198 return wxExecute(cmd) != 0;
199}
200
201void wxHandleProcessTermination(wxEndProcessData *proc_data)
202{
203 int pid = (proc_data->pid > 0) ? proc_data->pid : -(proc_data->pid);
204
205 // waitpid is POSIX so should be available everywhere, however on older
206 // systems wait() might be used instead in a loop (until the right pid
207 // terminates)
208 int status = 0;
209 if ( waitpid(pid, &status, 0) == -1 || !WIFEXITED(status) )
210 {
211 wxLogSysError(_("Waiting for subprocess termination failed"));
212 }
213 else
214 {
215 // notify user about termination if required
216 if (proc_data->process)
217 {
218 proc_data->process->OnTerminate(proc_data->pid,
219 WEXITSTATUS(status));
220 }
221 }
222
223 // clean up
224 if ( proc_data->pid > 0 )
225 {
226 delete proc_data;
227 }
228 else
229 {
230 // wxExecute() will know about it
231 proc_data->exitcode = status;
232
233 proc_data->pid = 0;
234 }
235}
236
237long wxExecute( wxChar **argv, bool sync, wxProcess *process )
238{
239 wxCHECK_MSG( *argv, 0, _T("can't exec empty command") );
240
241 int end_proc_detect[2];
242#if wxUSE_UNICODE
243 int mb_argc = 0;
244 char *mb_argv[WXEXECUTE_NARGS];
245
246 while (argv[mb_argc]) {
247 wxWX2MBbuf mb_arg = wxConv_libc.cWX2MB(argv[mb_argc]);
248 mb_argv[mb_argc] = strdup(mb_arg);
249 mb_argc++;
250 }
251 mb_argv[mb_argc] = (char *) NULL;
252#else
253 wxChar **mb_argv = argv;
254#endif
255
256 // create pipes
257 if (pipe(end_proc_detect) == -1)
258 {
259 wxLogSysError( _("Pipe creation failed") );
260#if wxUSE_UNICODE
261 mb_argc = 0;
262 while (mb_argv[mb_argc])
263 free(mb_argv[mb_argc++]);
264#endif
265 return 0;
266 }
267
268 // fork the process
269#ifdef HAVE_VFORK
270 pid_t pid = vfork();
271#else
272 pid_t pid = fork();
273#endif
274 if (pid == -1)
275 {
276 wxLogSysError( _("Fork failed") );
277#if wxUSE_UNICODE
278 mb_argc = 0;
279 while (mb_argv[mb_argc])
280 free(mb_argv[mb_argc++]);
281#endif
282 return 0;
283 }
284 else if (pid == 0)
285 {
286 // we're in child
287 close(end_proc_detect[0]); // close reading side
288
289 // These three lines close the open file descriptors to to avoid any
290 // input/output which might block the process or irritate the user. If
291 // one wants proper IO for the subprocess, the right thing to do is
292 // to start an xterm executing it.
293 if (sync == 0)
294 {
295 // leave stderr opened, it won't do any hurm
296 for ( int fd = 0; fd < FD_SETSIZE; fd++ )
297 {
298 if ( fd != end_proc_detect[1] && fd != STDERR_FILENO )
299 close(fd);
300 }
301 }
302
303#if 0
304 close(STDERR_FILENO);
305
306 // some programs complain about stderr not being open, so redirect
307 // them:
308 open("/dev/null", O_RDONLY); // stdin
309 open("/dev/null", O_WRONLY); // stdout
310 open("/dev/null", O_WRONLY); // stderr
311#endif
312
313 execvp (*mb_argv, mb_argv);
314
315 // there is no return after successful exec()
316 wxFprintf(stderr, _("Can't execute '%s'\n"), *argv);
317
318 _exit(-1);
319 }
320 else
321 {
322 // we're in parent
323 close(end_proc_detect[1]); // close writing side
324
325 wxEndProcessData *data = new wxEndProcessData;
326 data->tag = wxAddProcessCallback(data, end_proc_detect[0]);
327
328#if wxUSE_UNICODE
329 mb_argc = 0;
330 while (mb_argv[mb_argc])
331 free(mb_argv[mb_argc++]);
332#endif
333
334 if ( sync )
335 {
336 wxASSERT_MSG( !process, _T("wxProcess param ignored for sync exec") );
337 data->process = NULL;
338
339 // sync execution: indicate it by negating the pid
340 data->pid = -pid;
341
342 // it will be set to 0 from GTK_EndProcessDetector
343 while (data->pid != 0)
344 wxYield();
345
346 int exitcode = data->exitcode;
347
348 delete data;
349
350 return exitcode;
351 }
352 else
353 {
354 // async execution, nothing special to do - caller will be
355 // notified about the process terminationif process != NULL, data
356 // will be deleted in GTK_EndProcessDetector
357 data->process = process;
358 data->pid = pid;
359
360 return pid;
361 }
362 }
363}
364
365// ----------------------------------------------------------------------------
366// file and directory functions
367// ----------------------------------------------------------------------------
368
369const wxChar* wxGetHomeDir( wxString *home )
370{
371 *home = wxGetUserHome( wxString() );
372 if ( home->IsEmpty() )
373 *home = _T("/");
374
375 return home->c_str();
376}
377
378#if wxUSE_UNICODE
379const wxMB2WXbuf wxGetUserHome( const wxString &user )
380#else // just for binary compatibility
381char *wxGetUserHome( const wxString &user )
382#endif
383{
384 struct passwd *who = (struct passwd *) NULL;
385
386 if ( !user )
387 {
388 register wxChar *ptr;
389
390 if ((ptr = wxGetenv(_T("HOME"))) != NULL)
391 {
392 return ptr;
393 }
394 if ((ptr = wxGetenv(_T("USER"))) != NULL || (ptr = wxGetenv(_T("LOGNAME"))) != NULL)
395 {
396 who = getpwnam(wxConv_libc.cWX2MB(ptr));
397 }
398
399 // We now make sure the the user exists!
400 if (who == NULL)
401 {
402 who = getpwuid(getuid());
403 }
404 }
405 else
406 {
407 who = getpwnam (user.mb_str());
408 }
409
410#if wxUSE_UNICODE
411 return who ? wxConv_libc.cMB2WX(who->pw_dir) : (wxMB2WXbuf)((wxChar*)NULL);
412#else
413 return who ? who->pw_dir : ((char*)NULL);
414#endif
415}
416
417// ----------------------------------------------------------------------------
418// network and user id routines
419// ----------------------------------------------------------------------------
420
421// retrieve either the hostname or FQDN depending on platform (caller must
422// check whether it's one or the other, this is why this function is for
423// private use only)
424static bool wxGetHostNameInternal(wxChar *buf, int sz)
425{
426 wxCHECK_MSG( buf, FALSE, _T("NULL pointer in wxGetHostNameInternal") );
427
428 *buf = _T('\0');
429
430 // we're using uname() which is POSIX instead of less standard sysinfo()
431#if defined(HAVE_UNAME)
432 struct utsname uts;
433 bool ok = uname(&uts) != -1;
434 if ( ok )
435 {
436 wxStrncpy(buf, wxConv_libc.cMB2WX(uts.nodename), sz - 1);
437 buf[sz] = _T('\0');
438 }
439#elif defined(HAVE_GETHOSTNAME)
440 bool ok = gethostname(buf, sz) != -1;
441#else // no uname, no gethostname
442 wxFAIL_MSG(_T("don't know host name for this machine"));
443
444 bool ok = FALSE;
445#endif // uname/gethostname
446
447 if ( !ok )
448 {
449 wxLogSysError(_("Cannot get the hostname"));
450 }
451
452 return ok;
453}
454
455bool wxGetHostName(wxChar *buf, int sz)
456{
457 bool ok = wxGetHostNameInternal(buf, sz);
458
459 if ( ok )
460 {
461 // BSD systems return the FQDN, we only want the hostname, so extract
462 // it (we consider that dots are domain separators)
463 wxChar *dot = wxStrchr(buf, _T('.'));
464 if ( dot )
465 {
466 // nuke it
467 *dot = _T('\0');
468 }
469 }
470
471 return ok;
472}
473
474bool wxGetFullHostName(wxChar *buf, int sz)
475{
476 bool ok = wxGetHostNameInternal(buf, sz);
477
478 if ( ok )
479 {
480 if ( !wxStrchr(buf, _T('.')) )
481 {
482 struct hostent *host = gethostbyname(wxConv_libc.cWX2MB(buf));
483 if ( !host )
484 {
485 wxLogSysError(_("Cannot get the official hostname"));
486
487 ok = FALSE;
488 }
489 else
490 {
491 // the canonical name
492 wxStrncpy(buf, wxConv_libc.cMB2WX(host->h_name), sz);
493 }
494 }
495 //else: it's already a FQDN (BSD behaves this way)
496 }
497
498 return ok;
499}
500
501bool wxGetUserId(wxChar *buf, int sz)
502{
503 struct passwd *who;
504
505 *buf = _T('\0');
506 if ((who = getpwuid(getuid ())) != NULL)
507 {
508 wxStrncpy (buf, wxConv_libc.cMB2WX(who->pw_name), sz - 1);
509 return TRUE;
510 }
511
512 return FALSE;
513}
514
515bool wxGetUserName(wxChar *buf, int sz)
516{
517 struct passwd *who;
518 char *comma;
519
520 *buf = _T('\0');
521 if ((who = getpwuid (getuid ())) != NULL) {
522 comma = strchr(who->pw_gecos, ',');
523 if (comma)
524 *comma = '\0'; // cut off non-name comment fields
525 wxStrncpy (buf, wxConv_libc.cMB2WX(who->pw_gecos), sz - 1);
526 return TRUE;
527 }
528
529 return FALSE;
530}
531
532// ----------------------------------------------------------------------------
533// error and debug output routines (deprecated, use wxLog)
534// ----------------------------------------------------------------------------
535
536void wxDebugMsg( const char *format, ... )
537{
538 va_list ap;
539 va_start( ap, format );
540 vfprintf( stderr, format, ap );
541 fflush( stderr );
542 va_end(ap);
543}
544
545void wxError( const wxString &msg, const wxString &title )
546{
547 wxFprintf( stderr, _("Error ") );
548 if (!title.IsNull()) wxFprintf( stderr, _T("%s "), WXSTRINGCAST(title) );
549 if (!msg.IsNull()) wxFprintf( stderr, _T(": %s"), WXSTRINGCAST(msg) );
550 wxFprintf( stderr, _T(".\n") );
551}
552
553void wxFatalError( const wxString &msg, const wxString &title )
554{
555 wxFprintf( stderr, _("Error ") );
556 if (!title.IsNull()) wxFprintf( stderr, _T("%s "), WXSTRINGCAST(title) );
557 if (!msg.IsNull()) wxFprintf( stderr, _T(": %s"), WXSTRINGCAST(msg) );
558 wxFprintf( stderr, _T(".\n") );
559 exit(3); // the same exit code as for abort()
560}