]> git.saurik.com Git - wxWidgets.git/blame - src/gtk/utilsgtk.cpp
Sorry, I went and removed consts as per the style guide :-)
[wxWidgets.git] / src / gtk / utilsgtk.cpp
CommitLineData
c801d85f
KB
1/////////////////////////////////////////////////////////////////////////////
2// Name: utils.cpp
3// Purpose:
4// Author: Robert Roebling
5// Created: 01/02/97
6// Id:
7// Copyright: (c) 1998 Robert Roebling, Julian Smart and Markus Holzem
8// Licence: wxWindows licence
9/////////////////////////////////////////////////////////////////////////////
10
11
12//#ifdef __GNUG__
13//#pragma implementation "utils.h"
14//#endif
15
16#include "wx/utils.h"
17#include "wx/string.h"
18
19#include <stdarg.h>
20#include <dirent.h>
21#include <string.h>
22#include <sys/stat.h>
23#include <sys/types.h>
24#include <unistd.h>
25#include <sys/wait.h>
26#include <pwd.h>
27#include <errno.h>
28#include <netdb.h>
82052aff 29#include <signal.h>
c801d85f
KB
30
31#ifdef __SVR4__
32#include <sys/systeminfo.h>
33#endif
34
35//------------------------------------------------------------------------
36// misc.
37//------------------------------------------------------------------------
38
39void wxBell(void)
40{
41 gdk_beep();
42};
43
82052aff
GL
44void wxSleep(int nSecs)
45{
46 sleep(nSecs);
47};
48
49int wxKill(long pid, int sig)
50{
51 return kill(pid, sig);
52};
53
c801d85f
KB
54//------------------------------------------------------------------------
55// user and home routines
56//------------------------------------------------------------------------
57
58char* wxGetHomeDir( char *dest )
59{
60 wxString tmp = wxGetUserHome( wxString() );
61 if (tmp.IsNull())
62 strcpy( wxBuffer, "/" );
63 else
64 strcpy( wxBuffer, tmp );
65 if (dest) strcpy( dest, WXSTRINGCAST tmp );
66 return wxBuffer;
67};
68
69char *wxGetUserHome( const wxString &user )
70{
71 struct passwd *who = NULL;
72
73 if (user.IsNull() || (user== ""))
74 {
75 register char *ptr;
76
77 if ((ptr = getenv("HOME")) != NULL)
78 return ptr;
79 if ((ptr = getenv("USER")) != NULL
80 || (ptr = getenv("LOGNAME")) != NULL) {
81 who = getpwnam(ptr);
82 }
83 // We now make sure the the user exists!
84 if (who == NULL)
85 who = getpwuid(getuid());
86 }
87 else
88 who = getpwnam (user);
89
90 return who ? who->pw_dir : (char*)NULL;
91};
92
93//------------------------------------------------------------------------
94// id routines
95//------------------------------------------------------------------------
96
97bool wxGetHostName(char *buf, int sz)
98{
99 *buf = '\0';
100#if defined(__SVR4__) && !defined(__sgi)
101 return (sysinfo(SI_HOSTNAME, buf, sz) != -1);
102#else /* BSD Sockets */
103 char name[255];
104 struct hostent *h;
105 // Get hostname
106 if (gethostname(name, sizeof(name)/sizeof(char)-1) == -1)
107 return FALSE;
108 // Get official full name of host
109 strncpy(buf, (h=gethostbyname(name))!=NULL ? h->h_name : name, sz-1);
110 return TRUE;
111#endif
112}
113
114bool wxGetUserId(char *buf, int sz)
115{
116 struct passwd *who;
117
118 *buf = '\0';
119 if ((who = getpwuid(getuid ())) != NULL) {
120 strncpy (buf, who->pw_name, sz-1);
121 return TRUE;
122 }
123 return FALSE;
124}
125
126bool wxGetUserName(char *buf, int sz)
127{
128 struct passwd *who;
129
130 *buf = '\0';
131 if ((who = getpwuid (getuid ())) != NULL) {
132 strncpy (buf, who->pw_gecos, sz - 1);
133 return TRUE;
134 }
135 return FALSE;
136}
137
138//------------------------------------------------------------------------
139// error and debug output routines
140//------------------------------------------------------------------------
141
142void wxDebugMsg( const char *format, ... )
143{
144 va_list ap;
145 va_start( ap, format );
146 vfprintf( stderr, format, ap );
147 fflush( stderr );
148 va_end(ap);
149};
150
151void wxError( const wxString &msg, const wxString &title )
152{
153 fprintf( stderr, "Error " );
154 if (!title.IsNull()) fprintf( stderr, "%s ", WXSTRINGCAST(title) );
155 if (!msg.IsNull()) fprintf( stderr, ": %s", WXSTRINGCAST(msg) );
156 fprintf( stderr, ".\n" );
157};
158
159void wxFatalError( const wxString &msg, const wxString &title )
160{
161 fprintf( stderr, "Error " );
162 if (!title.IsNull()) fprintf( stderr, "%s ", WXSTRINGCAST(title) );
163 if (!msg.IsNull()) fprintf( stderr, ": %s", WXSTRINGCAST(msg) );
164 fprintf( stderr, ".\n" );
165 exit(1);
166};
167
168//------------------------------------------------------------------------
169// directory routines
170//------------------------------------------------------------------------
171
172bool wxDirExists( const wxString& dir )
173{
174 char buf[500];
175 strcpy( buf, WXSTRINGCAST(dir) );
176 struct stat sbuf;
177 return ((stat(buf, &sbuf) != -1) && S_ISDIR(sbuf.st_mode) ? TRUE : FALSE);
178};
179
180//------------------------------------------------------------------------
181// wild character routines
182//------------------------------------------------------------------------
183
184bool wxIsWild( const wxString& pattern )
185{
186 wxString tmp = pattern;
187 char *pat = WXSTRINGCAST(tmp);
188 while (*pat) {
189 switch (*pat++) {
190 case '?': case '*': case '[': case '{':
191 return TRUE;
192 case '\\':
193 if (!*pat++)
194 return FALSE;
195 }
196 }
197 return FALSE;
198};
199
200
201bool wxMatchWild( const wxString& pat, const wxString& text, bool dot_special )
202{
203 wxString tmp1 = pat;
204 char *pattern = WXSTRINGCAST(tmp1);
205 wxString tmp2 = text;
206 char *str = WXSTRINGCAST(tmp2);
207 char c;
208 char *cp;
209 bool done = FALSE, ret_code, ok;
210 // Below is for vi fans
211 const char OB = '{', CB = '}';
212
213 // dot_special means '.' only matches '.'
214 if (dot_special && *str == '.' && *pattern != *str)
215 return FALSE;
216
217 while ((*pattern != '\0') && (!done)
218 && (((*str=='\0')&&((*pattern==OB)||(*pattern=='*')))||(*str!='\0'))) {
219 switch (*pattern) {
220 case '\\':
221 pattern++;
222 if (*pattern != '\0')
223 pattern++;
224 break;
225 case '*':
226 pattern++;
227 ret_code = FALSE;
228 while ((*str!='\0')
229 && (!(ret_code=wxMatchWild(pattern, str++, FALSE))))
230 /*loop*/;
231 if (ret_code) {
232 while (*str != '\0')
233 str++;
234 while (*pattern != '\0')
235 pattern++;
236 }
237 break;
238 case '[':
239 pattern++;
240 repeat:
241 if ((*pattern == '\0') || (*pattern == ']')) {
242 done = TRUE;
243 break;
244 }
245 if (*pattern == '\\') {
246 pattern++;
247 if (*pattern == '\0') {
248 done = TRUE;
249 break;
250 }
251 }
252 if (*(pattern + 1) == '-') {
253 c = *pattern;
254 pattern += 2;
255 if (*pattern == ']') {
256 done = TRUE;
257 break;
258 }
259 if (*pattern == '\\') {
260 pattern++;
261 if (*pattern == '\0') {
262 done = TRUE;
263 break;
264 }
265 }
266 if ((*str < c) || (*str > *pattern)) {
267 pattern++;
268 goto repeat;
269 }
270 } else if (*pattern != *str) {
271 pattern++;
272 goto repeat;
273 }
274 pattern++;
275 while ((*pattern != ']') && (*pattern != '\0')) {
276 if ((*pattern == '\\') && (*(pattern + 1) != '\0'))
277 pattern++;
278 pattern++;
279 }
280 if (*pattern != '\0') {
281 pattern++, str++;
282 }
283 break;
284 case '?':
285 pattern++;
286 str++;
287 break;
288 case OB:
289 pattern++;
290 while ((*pattern != CB) && (*pattern != '\0')) {
291 cp = str;
292 ok = TRUE;
293 while (ok && (*cp != '\0') && (*pattern != '\0')
294 && (*pattern != ',') && (*pattern != CB)) {
295 if (*pattern == '\\')
296 pattern++;
297 ok = (*pattern++ == *cp++);
298 }
299 if (*pattern == '\0') {
300 ok = FALSE;
301 done = TRUE;
302 break;
303 } else if (ok) {
304 str = cp;
305 while ((*pattern != CB) && (*pattern != '\0')) {
306 if (*++pattern == '\\') {
307 if (*++pattern == CB)
308 pattern++;
309 }
310 }
311 } else {
312 while (*pattern!=CB && *pattern!=',' && *pattern!='\0') {
313 if (*++pattern == '\\') {
314 if (*++pattern == CB || *pattern == ',')
315 pattern++;
316 }
317 }
318 }
319 if (*pattern != '\0')
320 pattern++;
321 }
322 break;
323 default:
324 if (*str == *pattern) {
325 str++, pattern++;
326 } else {
327 done = TRUE;
328 }
329 }
330 }
331 while (*pattern == '*')
332 pattern++;
333 return ((*str == '\0') && (*pattern == '\0'));
334};
335
336//------------------------------------------------------------------------
337// subprocess routines
338//------------------------------------------------------------------------
339
cf447356
GL
340typedef struct {
341 gint pid, tag;
342 wxProcess *process;
343} wxEndProcessData;
344
345static void GTK_EndProcessDetector(gpointer data, gint source,
346 GdkInputCondition condition)
347{
348 wxEndProcessData *proc_data = (wxEndProcessData *)data;
349 int pid;
350
351 pid = (proc_data->pid > 0) ? proc_data->pid : -(proc_data->pid);
352
353 wait4(proc_data->pid, NULL, 0, NULL);
354
355 close(source);
356 gdk_input_remove(proc_data->tag);
357
358 if (proc_data->process)
359 proc_data->process->OnTerminate(proc_data->pid);
360
361 if (proc_data->pid > 0)
362 delete proc_data;
363 else
364 proc_data->pid = 0;
365};
366
eafc087e 367long wxExecute( char **argv, bool sync, wxProcess *process )
c801d85f 368{
cf447356
GL
369 wxEndProcessData *data = new wxEndProcessData;
370 int end_proc_detect[2];
371
c801d85f 372 if (*argv == NULL)
cf447356
GL
373 return 0;
374
375 /* Create pipes */
376 if (pipe(end_proc_detect) == -1) {
377 perror("pipe failed");
378 return 0;
379 }
c801d85f
KB
380
381 /* fork the process */
382#if defined(sun) || defined(__ultrix) || defined(__bsdi__)
383 pid_t pid = vfork();
384#else
385 pid_t pid = fork();
386#endif
387 if (pid == -1) {
388 perror ("fork failed");
cf447356 389 return 0;
c801d85f 390 } else if (pid == 0) {
cf447356
GL
391 /* Close fd not useful */
392 close(end_proc_detect[0]); // close reading side
393
c801d85f
KB
394 /* child */
395#ifdef _AIX
396 execvp ((const char *)*argv, (const char **)argv);
397#else
398 execvp (*argv, argv);
399#endif
400 if (errno == ENOENT)
401 wxError("command not found", *argv);
402 else
403 perror (*argv);
404 wxError("could not execute", *argv);
405 _exit (-1);
406 }
407
cf447356
GL
408 close(end_proc_detect[1]); // close writing side
409 data->tag = gdk_input_add(end_proc_detect[0], GDK_INPUT_READ,
410 GTK_EndProcessDetector, (gpointer)data);
411 data->pid = pid;
eafc087e 412 if (!sync) {
cf447356
GL
413 data->process = process;
414 } else {
415 data->process = NULL;
416 data->pid = -(data->pid);
417
418 while (data->pid != 0)
419 wxYield();
420
421 delete data;
422 }
423
424 return pid;
c801d85f
KB
425};
426
eafc087e 427long wxExecute( const wxString& command, bool sync, wxProcess *process )
c801d85f
KB
428{
429 if (command.IsNull() || command == "") return FALSE;
430
431 int argc = 0;
432 char *argv[127];
433 char tmp[1024];
434 const char *IFS = " \t\n";
435
436 strncpy (tmp, command, sizeof(tmp) / sizeof(char) - 1);
437 tmp[sizeof (tmp) / sizeof (char) - 1] = '\0';
438 argv[argc++] = strtok (tmp, IFS);
439 while ((argv[argc++] = strtok(NULL, IFS)) != NULL)
440 /* loop */ ;
eafc087e 441 return wxExecute(argv, sync, process);
c801d85f
KB
442};
443