source: trunk/dll/systemf.c@ 985

Last change on this file since 985 was 985, checked in by Gregg Young, 18 years ago

Update sizes dialog (ticket 44); Make max command line length user settable (ticket 199); use xfree for free in most cases (ticket 212); initial code to check for valid ini file (ticket 102); Some additional refactoring and structure rework; Some documentation updates;

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 30.8 KB
Line 
1
2/***********************************************************************
3
4 $Id: systemf.c 985 2008-03-01 01:37:14Z gyoung $
5
6 System Interfaces
7
8 Copyright (c) 1993-98 M. Kimes
9 Copyright (c) 2003, 2006 Steven H.Levine
10
11 21 Nov 03 SHL Comments
12 31 Jul 04 SHL Indent -i2
13 01 Aug 04 SHL Rework lstrip/rstrip usage
14 17 Jul 06 SHL Use Runtime_Error
15 26 Jul 06 SHL Use convert_nl_to_nul
16 15 Aug 06 SHL More error popups
17 01 Nov 06 SHL runemf2: temp fix for hung windows caused by termq errors
18 03 Nov 06 SHL runemf2: rework termination queue logic to work for multiple threads
19 07 Jan 07 GKY Move error strings etc. to string file
20 06 Aug 07 GKY Reduce DosSleep times (ticket 148)
21 20 Aug 07 GKY Move #pragma alloc_text to end for OpenWatcom compat
22 29 Feb 08 GKY Use xfree where appropriate
23 29 Feb 08 GKY Changes to enable user settable command line length
24 29 Feb 08 GKY Refactor global command line variables to notebook.h
25
26***********************************************************************/
27
28#include <stdlib.h>
29#include <stdarg.h>
30#include <string.h>
31#include <ctype.h>
32
33#define INCL_DOS
34#define INCL_DOSERRORS
35#define INCL_WIN
36#define INCL_LONGLONG // dircnrs.h
37
38#include "fm3dlg.h"
39#include "fm3str.h"
40#include "errutil.h" // Dos_Error...
41#include "strutil.h" // GetPString
42#include "notebook.h" //targetdirectory
43#include "pathutil.h"
44#include "fm3dll.h"
45
46static PSZ pszSrcFile = __FILE__;
47
48/**
49 * Bring session foreground
50 * @return TRUE if OK, else FALSE
51 */
52
53BOOL ShowSession(HWND hwnd, PID pid)
54{
55 HSWITCH hswitch;
56 SWCNTRL swctl;
57 ULONG rc;
58
59 hswitch = WinQuerySwitchHandle(pid ? (HWND)0 : hwnd, pid);
60 if (hswitch) {
61 rc = WinQuerySwitchEntry(hswitch, &swctl);
62 if (!rc) {
63 if (swctl.idProcess == pid && swctl.uchVisibility == SWL_VISIBLE)
64 rc = WinSwitchToProgram(hswitch);
65 if (!rc)
66 return TRUE;
67 // else saymsg(MB_ENTER,HWND_DESKTOP,DEBUG_STRING,"Failed: %lu/%lx",rc,rc);
68
69 }
70 }
71 return FALSE;
72}
73
74/**
75 * Invoke runemf2 for command and file/directory list
76 * @return command return code or
77 * -1 if runtime error or
78 * -2 if user cancels command line edit dialog
79 */
80
81int ExecOnList(HWND hwnd, char *command, int flags, char *tpath,
82 char **list, char *prompt, PCSZ pszCallingFile, UINT uiLineNumber)
83{
84 /* executes the command once for all files in list */
85
86 char path[CCHMAXPATH], *commandline, modpath[CCHMAXPATH], listfile[CCHMAXPATH],
87 *p, *pp, drive, *file, *ext, *dot;
88 register int x;
89 BOOL spaces;
90
91 if (!command || !*command) {
92 Runtime_Error2(pszSrcFile, __LINE__, IDS_NODATATEXT);
93 return -1;
94 }
95 commandline = xmalloc(MaxComLineStrg, pszSrcFile, __LINE__);
96 if (!commandline)
97 return -1; //already complained
98 *listfile = 0;
99 bstrip(command);
100
101 *path = 0;
102 if (tpath && *tpath)
103 strcpy(path, tpath);
104 else if (*command != '<' || !strchr(command, '>')) {
105 strcpy(path, command + (*command == '"'));
106 if (*command == '\"')
107 p = strchr(path, '\"');
108 else
109 p = strchr(path, ' ');
110 if (p)
111 *p = 0;
112 p = strrchr(path, '\\');
113 if (!p)
114 p = strrchr(path, ':');
115 if (p) {
116 if (*p == ':') {
117 p++;
118 *p = '\\';
119 p++;
120 }
121 *p = 0;
122 }
123 else
124 *path = 0;
125 }
126 if (!*path) {
127 if (list && list[0])
128 strcpy(path, list[0]);
129 p = strrchr(path, '\\');
130 if (!p)
131 p = strrchr(path, ':');
132 if (p) {
133 if (*p == ':') {
134 p++;
135 *p = '\\';
136 p++;
137 }
138 *p = 0;
139 }
140 else
141 *path = 0;
142 }
143 *modpath = 0;
144 if (list && list[0])
145 strcpy(modpath, list[0]);
146 p = strrchr(modpath, '\\');
147 if (!p)
148 p = strrchr(modpath, ':');
149 if (p) {
150 if (*p == ':') {
151 p++;
152 *p = '\\';
153 p++;
154 }
155 *p = 0;
156 }
157 else
158 *modpath = 0;
159 if (!*modpath)
160 strcpy(modpath, path);
161 if (*path)
162 MakeFullName(path);
163 if (*modpath)
164 MakeFullName(modpath);
165 if (IsFullName(path))
166 drive = toupper(*path);
167 else
168 drive = 0;
169
170 p = command; // substitue for special % sequences
171
172 pp = commandline;
173 *commandline = 0;
174 while (*p) {
175 if (*p == '%') {
176 switch (*(p + 1)) {
177 case '!': /* write list to file, add filename */
178 if (list) {
179 if (!*listfile) {
180 FILE *fp;
181
182 save_dir2(listfile);
183 if (listfile[strlen(listfile) - 1] != '\\')
184 strcat(listfile, "\\");
185 sprintf(&listfile[strlen(listfile)], "%s%03x",
186 LISTTEMPROOT, (clock() & 4095));
187 fp = xfopen(listfile, "w",pszSrcFile,__LINE__);
188 if (fp) {
189 for (x = 0; list[x]; x++)
190 {
191 fputs(list[x], fp);
192 if (list[x + 1])
193 fputc('\n', fp);
194 }
195 fclose(fp);
196 }
197 }
198 strcpy(pp, listfile);
199 pp += strlen(listfile);
200 }
201 p += 2;
202 break;
203
204 case 'c': /* add name of command processor */
205 {
206 char *env = GetCmdSpec(FALSE);
207
208 if (needs_quoting(env) && !strchr(env, '\"')) {
209 *pp = '\"';
210 pp++;
211 spaces = TRUE;
212 }
213 else
214 spaces = FALSE;
215 strcpy(pp, env);
216 p += 2;
217 pp += strlen(env);
218 if (spaces) {
219 *pp = '\"';
220 pp++;
221 }
222 }
223 break;
224
225 case 't': /* add Target directory */
226 if (needs_quoting(targetdir) && !strchr(targetdir, '\"')) {
227 *pp = '\"';
228 pp++;
229 spaces = TRUE;
230 }
231 else
232 spaces = FALSE;
233 strcpy(pp, targetdir);
234 p += 2;
235 pp += strlen(targetdir);
236 if (spaces) {
237 *pp = '\"';
238 pp++;
239 }
240 break;
241
242 case '$': /* add drive letter */
243 if (drive)
244 *pp = drive;
245 else {
246 ULONG ulDriveNum = 3, ulDriveMap;
247
248 DosQCurDisk(&ulDriveNum, &ulDriveMap);
249 *pp = (char) (ulDriveNum + '@');
250 }
251 pp++;
252 p += 2;
253 break;
254
255 case 'U': /* add path of first list component */
256 case 'u':
257 if (*modpath) {
258 if (needs_quoting(modpath) && !strchr(modpath, '\"')) {
259 spaces = TRUE;
260 *pp = '\"';
261 pp++;
262 }
263 else
264 spaces = FALSE;
265 if (*(p + 1) == 'u') {
266 strcpy(pp, modpath);
267 pp += strlen(modpath);
268 }
269 else {
270 strcpy(pp, modpath + 2);
271 pp += strlen(modpath + 2);
272 }
273 if (spaces) {
274 if (modpath[strlen(modpath) - 1] == '\\') {
275 *pp = '\\';
276 pp++;
277 }
278 *pp = '\"';
279 pp++;
280 }
281 }
282 else {
283 char temp[CCHMAXPATH];
284
285 save_dir2(temp);
286 if (needs_quoting(temp) && !strchr(temp, '\"')) {
287 spaces = TRUE;
288 *pp = '\"';
289 pp++;
290 }
291 else
292 spaces = FALSE;
293 strcpy(pp, temp);
294 pp += strlen(temp);
295 if (spaces) {
296 if (temp[strlen(temp) - 1] == '\\') {
297 *pp = '\\';
298 pp++;
299 }
300 *pp = '\"';
301 pp++;
302 }
303 }
304 p += 2;
305 break;
306
307 case 'P': /* add path of execution */
308 case 'p':
309 if (*path) {
310 if (needs_quoting(path) && !strchr(path, '\"')) {
311 spaces = TRUE;
312 *pp = '\"';
313 pp++;
314 }
315 else
316 spaces = FALSE;
317 if (*(p + 1) == 'p') {
318 strcpy(pp, path);
319 pp += strlen(path);
320 }
321 else {
322 strcpy(pp, path + 2);
323 pp += strlen(path + 2);
324 }
325 if (spaces) {
326 if (path[strlen(path) - 1] == '\\') {
327 *pp = '\\';
328 pp++;
329 }
330 *pp = '\"';
331 pp++;
332 }
333 }
334 else {
335 char temp[CCHMAXPATH];
336
337 save_dir2(temp);
338 if (needs_quoting(temp) && !strchr(temp, '\"')) {
339 spaces = TRUE;
340 *pp = '\"';
341 pp++;
342 }
343 else
344 spaces = FALSE;
345 strcpy(pp, temp);
346 pp += strlen(temp);
347 if (spaces) {
348 if (temp[strlen(temp) - 1] == '\\') {
349 *pp = '\\';
350 pp++;
351 }
352 *pp = '\"';
353 pp++;
354 }
355 }
356 p += 2;
357 break;
358
359 case 'D':
360 if (hwndMain) {
361 PCNRITEM pci;
362
363 pci = (PCNRITEM) WinSendMsg(WinWindowFromID(WinWindowFromID(
364 hwndTree, FID_CLIENT), TREE_CNR),
365 CM_QUERYRECORDEMPHASIS,
366 MPFROMLONG(CMA_FIRST),
367 MPFROMSHORT(CRA_CURSORED));
368 if (pci && (int) pci != -1 && *pci->pszFileName) {
369 if (needs_quoting(pci->pszFileName) &&
370 !strchr(pci->pszFileName, '\"'))
371 {
372 *pp = '\"';
373 pp++;
374 spaces = TRUE;
375 }
376 else
377 spaces = FALSE;
378 strcpy(pp, pci->pszFileName);
379 pp += strlen(pci->pszFileName);
380 if (spaces) {
381 *pp = '\"';
382 pp++;
383 }
384 }
385 }
386 p += 2;
387 break;
388
389 case 'd':
390 if (hwndMain) {
391 HENUM henum;
392 char retstr[CCHMAXPATH];
393 HWND hwndC, hwndDir;
394 USHORT id;
395 BOOL first = TRUE;
396
397 henum = WinBeginEnumWindows(hwndMain);
398 while ((hwndC = WinGetNextWindow(henum)) != NULLHANDLE) {
399 if (hwndC != hwndTree) {
400 id = WinQueryWindowUShort(hwndC, QWS_ID);
401 if (id) {
402 hwndDir = WinWindowFromID(hwndC, FID_CLIENT);
403 if (hwndDir) {
404 hwndDir = WinWindowFromID(hwndDir, DIR_CNR);
405 if (hwndDir) {
406 *retstr = 0;
407 WinSendMsg(hwndC, UM_CONTAINERDIR, MPFROMP(retstr), MPVOID);
408 if (*retstr) {
409 if (!first) {
410 *pp = ' ';
411 pp++;
412 }
413 first = FALSE;
414 if (needs_quoting(retstr) && !strchr(retstr, '\"')) {
415 *pp = '\"';
416 pp++;
417 spaces = TRUE;
418 }
419 else
420 spaces = FALSE;
421 strcpy(pp, retstr);
422 pp += strlen(retstr);
423 if (spaces) {
424 *pp = '\"';
425 pp++;
426 }
427 }
428 }
429 }
430 }
431 }
432 }
433 WinEndEnumWindows(henum);
434 }
435 p += 2;
436 break;
437
438 case '%':
439 *pp = '%';
440 pp++;
441 p += 2;
442 break;
443
444 case 'R':
445 case 'F':
446 case 'A':
447 case 'r':
448 case 'f':
449 case 'a':
450 case 'e':
451 if (list) {
452 for (x = 0; list[x]; x++)
453 {
454 file = strrchr(list[x], '\\');
455 if (!file)
456 file = strrchr(list[x], ':');
457 if (file)
458 file++;
459 else
460 file = list[x];
461 ext = strrchr(file, '.');
462 dot = ext;
463 if (ext)
464 ext++;
465 switch (*(p + 1)) {
466 case 'R':
467 case 'r':
468 if (pp + strlen(list[x]) > commandline + 1250)
469 goto BreakOut;
470 if (*(p + 1) == 'r') {
471 strcpy(pp, list[x]);
472 pp += strlen(list[x]);
473 }
474 else {
475 strcpy(pp, list[x] + 2);
476 pp += strlen(list[x] + 2);
477 }
478 break;
479
480 case 'F':
481 case 'f':
482 if (*(p + 1) == 'F' && dot)
483 *dot = 0;
484 if (pp + strlen(file) > commandline + 1250)
485 goto BreakOut;
486 if (needs_quoting(file)) {
487 spaces = TRUE;
488 *pp = '\"';
489 pp++;
490 }
491 else
492 spaces = FALSE;
493 strcpy(pp, file);
494 pp += strlen(file);
495 if (*(p + 1) == 'F' && dot)
496 *dot = '.';
497 if (spaces) {
498 if (*(pp - 1) != '\"') {
499 *pp = '\"';
500 pp++;
501 }
502 }
503 break;
504
505 case 'A':
506 case 'a':
507 if (pp + strlen(list[x]) > commandline + 1250)
508 goto BreakOut;
509 if (needs_quoting(list[x]) && !strchr(list[x], '\"')) {
510 spaces = TRUE;
511 *pp = '\"';
512 pp++;
513 }
514 else
515 spaces = FALSE;
516 if (*(p + 1) == 'a') {
517 strcpy(pp, list[x]);
518 pp += strlen(list[x]);
519 }
520 else {
521 strcpy(pp, list[x] + 2);
522 pp += strlen(list[x] + 2);
523 }
524 if (spaces) {
525 if (list[x][strlen(list[x]) - 1] == '\\') {
526 *pp = '\\';
527 pp++;
528 }
529 *pp = '\"';
530 pp++;
531 }
532 break;
533
534 case 'e':
535 if (ext) {
536 if (pp + strlen(ext) > commandline + 1250)
537 goto BreakOut;
538 if (needs_quoting(ext)) {
539 spaces = TRUE;
540 *pp = '\"';
541 pp++;
542 }
543 else
544 spaces = FALSE;
545 strcpy(pp, ext);
546 pp += strlen(ext);
547 if (spaces) {
548 if (*(pp - 1) != '\"') {
549 *pp = '\"';
550 pp++;
551 }
552 }
553 }
554 break;
555 }
556 if (list[x + 1]) {
557 *pp = ' ';
558 pp++;
559 }
560 }
561 }
562 p += 2;
563 break;
564
565 default:
566 *pp = *p;
567 p++;
568 pp++;
569 break;
570 }
571 }
572 else {
573 *pp = *p;
574 pp++;
575 p++;
576 }
577 *pp = 0;
578 }
579
580BreakOut:
581
582 {
583 EXECARGS ex;
584 ULONG size;
585 int ret;
586
587 memset(&ex, 0, sizeof(EXECARGS));
588 size = sizeof(ex.environment) - 1;
589 PrfQueryProfileData(fmprof, FM3Str, command, ex.environment, &size);
590 if (flags & PROMPT) {
591 /* allow editing command line */
592 ex.flags = (flags & (~PROMPT));
593 ex.commandline = commandline;
594 strcpy(ex.path, path);
595 if (prompt)
596 strcpy(ex.title, prompt);
597 ret = WinDlgBox(HWND_DESKTOP, hwnd, CmdLineDlgProc, FM3ModHandle,
598 EXEC_FRAME, &ex);
599 if (ret != 1) {
600 xfree(commandline);
601 return (ret == 0) ? -1 : -2;
602 }
603 }
604 else
605 ex.flags = flags;
606 ex.flags &= (~PROMPT);
607 return runemf2(ex.flags, hwnd, pszCallingFile, uiLineNumber, path,
608 (*ex.environment) ? ex.environment : NULL,
609 "%s", commandline);
610 }
611}
612
613/** Run requested app
614 * @return application return code or -1 if problem starting app
615 */
616
617int runemf2(int type, HWND hwnd, PCSZ pszCallingFile, UINT uiLineNumber,
618 char *pszDirectory, char *pszEnvironment,
619 char *formatstring,...)
620{
621 /** example:
622
623 * status = runemf2(SEPARATE | WINDOWED,
624 * hwnd, pszCallingFile, __LINE__,
625 * NullStr,
626 * NULL,
627 * "%s /C %s",
628 * getenv("COMSPEC"),
629 * batchfilename);
630 *
631 * use (HWND)0 for hwnd if window handle not handy.
632 * pszCallingFile and __LINE__ are used to determine caller for easier error tracking
633 */
634
635 /**
636 * type bitmapped flag -- see FM3DLL.H
637 */
638
639 va_list parguments;
640 int ret = -1;
641 RESULTCODES results;
642 STARTDATA sdata;
643 REQUESTDATA rq;
644 ULONG ulSessID;
645 ULONG ulLength;
646 UINT ctr;
647 ULONG ulAppType;
648 PID sessPID;
649 BOOL wasquote;
650 char *pszPgm, *pszArgs = NULL;
651 char szObject[32] = "", *p, szSavedir[CCHMAXPATH];
652 BOOL useTermQ = FALSE;
653 char szTempdir[CCHMAXPATH];
654
655 typedef struct {
656 USHORT usSessID;
657 USHORT usRC;
658 } TERMINFO;
659
660 TERMINFO *pTermInfo;
661 BYTE bPriority;
662 APIRET rc;
663 PIB *ppib;
664 TIB *ptib;
665
666 // Shared by all threads
667# define TERMQ_BASE_NAME "\\QUEUES\\FM3WAIT"
668 static char szTermQName[30];
669 static HQUEUE hTermQ;
670 static HEV hTermQSem;
671
672 if (pszDirectory && *pszDirectory) {
673 if (!DosQueryPathInfo(pszDirectory,
674 FIL_QUERYFULLNAME,
675 szTempdir,
676 sizeof(szTempdir)))
677 pszDirectory = szTempdir;
678 }
679
680 if (!hwnd)
681 hwnd = HWND_DESKTOP;
682
683 rc = DosAllocMem((PVOID)&pszPgm,
684 MaxComLineStrg,
685 PAG_COMMIT | OBJ_TILE | PAG_READ | PAG_WRITE);
686 if (rc) {
687 Dos_Error(MB_CANCEL,rc,hwnd,pszSrcFile,__LINE__,GetPString(IDS_OUTOFMEMORY));
688 return -1;
689 }
690
691 *szSavedir = 0;
692
693 *pszPgm = 0;
694 va_start(parguments,
695 formatstring);
696 vsprintf(pszPgm,
697 formatstring,
698 parguments);
699 va_end(parguments);
700
701 if (pszEnvironment) {
702 p = &pszEnvironment[strlen(pszEnvironment)] + 1;
703 *p = 0;
704 p = pszEnvironment;
705 while ((p = convert_nl_to_nul(p)) != NULL)
706 ; // loop
707 }
708
709 if (!*pszPgm) {
710 p = GetCmdSpec(FALSE);
711 strcpy(pszPgm, p);
712 if (!*pszPgm) {
713 Runtime_Error2(pszSrcFile, __LINE__, IDS_NODATATEXT);
714 return -1;
715 }
716 }
717
718 if (*pszPgm) {
719 if (*pszPgm == '<' && strchr(pszPgm, '>')) {
720 /* is a workplace object */
721 HOBJECT hWPSObject;
722 char temp;
723
724 p = strchr(pszPgm, '>');
725 p++;
726 temp = *p;
727 if (temp) {
728 rc = DosAllocMem((PVOID)&pszArgs,
729 MaxComLineStrg * 2,
730 PAG_COMMIT | OBJ_TILE | PAG_READ | PAG_WRITE);
731 if (rc)
732 Dos_Error(MB_CANCEL,rc,hwnd,pszSrcFile,__LINE__,GetPString(IDS_OUTOFMEMORY));
733 }
734 else
735 pszArgs = NULL;
736 *p = 0;
737 /* Find the handle of the WPS object */
738 hWPSObject = WinQueryObject(pszPgm);
739 *p = temp;
740 if (hWPSObject != NULLHANDLE) {
741 if (pszArgs && *p) {
742 sprintf(pszArgs,"OPEN=DEFAULT;PARAMETERS=\"%s\"",p);
743 WinSetObjectData(hWPSObject,pszArgs);
744 }
745 else
746 WinSetObjectData(hWPSObject,"OPEN=DEFAULT");
747 ret = 0;
748 }
749 goto ObjectInterrupt;
750 }
751
752 if ((type & RUNTYPE_MASK) == SYNCHRONOUS ||
753 (type & RUNTYPE_MASK) == ASYNCHRONOUS ||
754 (type & RUNTYPE_MASK) == DETACHED)
755 {
756 strip_lead_char(" \t", pszPgm);
757 p = pszPgm;
758 wasquote = FALSE;
759 while (*p &&
760 (wasquote ||
761 (*p != ' ' &&
762 *p != '\t')))
763 {
764 if (*p == '\"') {
765 if (!wasquote) {
766 wasquote = TRUE;
767 memmove(p,
768 p + 1,
769 strlen(p));
770 while (*p == ' ' ||
771 *p == '\t')
772 p++;
773 }
774 else {
775 memmove(p,
776 p + 1,
777 strlen(p));
778 break;
779 }
780 }
781 else
782 p++;
783 }
784 if (*p) {
785 *p = 0;
786 p++;
787 }
788 else
789 p = pszPgm;
790 p[strlen(p) + 1] = 0; /* double-terminate args */
791 if (*pszPgm) {
792 if (!strchr(pszPgm, '\\') &&
793 !strchr(pszPgm, ':') &&
794 pszDirectory &&
795 *pszDirectory)
796 {
797 save_dir2(szSavedir);
798 switch_to(pszDirectory);
799 }
800 rc = DosQueryAppType(pszPgm,&ulAppType);
801 if (!strchr(pszPgm, '\\') &&
802 !strchr(pszPgm, ':') &&
803 pszDirectory &&
804 *pszDirectory)
805 switch_to(szSavedir);
806 if (rc) {
807 Dos_Error(MB_CANCEL,rc,hwnd,pszSrcFile,__LINE__,
808 GetPString(IDS_DOSQAPPTYPEFAILEDTEXT),
809 pszPgm, pszCallingFile, __LINE__);
810 DosFreeMem(pszPgm);
811 if (pszArgs)
812 DosFreeMem(pszArgs);
813 return -1;
814 }
815 if (ulAppType) {
816 if (ulAppType & FAPPTYP_DLL || ulAppType & FAPPTYP_VIRTDRV ||
817 ulAppType & FAPPTYP_PHYSDRV || ulAppType & FAPPTYP_PROTDLL)
818 {
819 Runtime_Error(pszSrcFile, __LINE__,
820 GetPString(IDS_APPTYPEUNEXPECTEDTEXT),
821 ulAppType, pszPgm, pszCallingFile, __LINE__);
822 if (pszPgm)
823 DosFreeMem(pszPgm);
824 if (pszArgs)
825 DosFreeMem(pszArgs);
826 return -1;
827 }
828 if (ulAppType & FAPPTYP_DOS || ulAppType & FAPPTYP_WINDOWSREAL ||
829 ulAppType & FAPPTYP_WINDOWSPROT || ulAppType & FAPPTYP_WINDOWSPROT31)
830 {
831 Runtime_Error(pszSrcFile, __LINE__,
832 GetPString(IDS_APPTYPEUNEXPECTEDTEXT),
833 ulAppType, pszPgm, pszCallingFile, __LINE__);
834 if (pszPgm)
835 DosFreeMem(pszPgm);
836 if (pszArgs)
837 DosFreeMem(pszArgs);
838 return -1;
839 }
840 }
841 memset(&results, 0, sizeof(results));
842 if (pszDirectory && *pszDirectory) {
843 save_dir2(szSavedir);
844 switch_to(pszDirectory);
845 }
846 ret = DosExecPgm(szObject, sizeof(szObject),
847 ((type & RUNTYPE_MASK) == ASYNCHRONOUS ? EXEC_ASYNC : 0) +
848 ((type & RUNTYPE_MASK) == DETACHED ? EXEC_BACKGROUND : 0),
849 pszPgm, pszEnvironment, &results, pszPgm);
850 if (pszDirectory && *pszDirectory)
851 switch_to(szSavedir);
852 if (ret) {
853 Dos_Error(MB_ENTER,ret,hwnd,pszSrcFile,__LINE__,
854 GetPString(IDS_DOSEXECPGMFAILEDTEXT), pszPgm,
855 pszCallingFile, __LINE__);
856 }
857 }
858 }
859 else {
860 if (~type & FULLSCREEN)
861 type |= WINDOWED;
862 rc = DosAllocMem((PVOID) & pszArgs, MaxComLineStrg * 2,
863 PAG_COMMIT | OBJ_TILE | PAG_READ | PAG_WRITE);
864 if (rc) {
865 Dos_Error(MB_CANCEL,rc,hwnd,pszSrcFile,__LINE__,GetPString(IDS_OUTOFMEMORY));
866 DosFreeMem(pszPgm);
867 return -1;
868 }
869 *pszArgs = 0;
870 memset(&sdata, 0, sizeof(sdata));
871 strip_lead_char(" \t", pszPgm);
872 p = pszPgm;
873 wasquote = FALSE;
874 while (*p && (wasquote || (*p != ' ' && *p != '\t'))) {
875 if (*p == '\"') {
876 if (!wasquote) {
877 wasquote = TRUE;
878 memmove(p, p + 1, strlen(p));
879 while (*p == ' ' || *p == '\t')
880 p++;
881 }
882 else {
883 memmove(p, p + 1, strlen(p));
884 break;
885 }
886 }
887 else
888 p++;
889 } // while
890 if (*p) {
891 *p = 0;
892 p++;
893 }
894 else
895 p = NullStr;
896 if (*p)
897 strcpy(pszArgs, p);
898
899 p = strrchr(pszPgm, '.');
900 if (p) {
901 char temp[CCHMAXPATH + 1];
902
903 if (!stricmp(p, ".BAT")) {
904 strcpy(temp, pszPgm);
905 strcpy(pszPgm, pszArgs);
906 strcpy(pszArgs, "/C ");
907 strcat(pszArgs, temp);
908 strcat(pszArgs, " ");
909 strcat(pszArgs, pszPgm);
910 strcpy(pszPgm, GetCmdSpec(TRUE)); // DOS
911 }
912 else if (!stricmp(p, ".CMD") || !stricmp(p, ".BTM")) {
913 // Assume 4OS2 is BTM
914 strcpy(temp, pszPgm);
915 strcpy(pszPgm, pszArgs);
916 strcpy(pszArgs, "/C ");
917 strcat(pszArgs, temp);
918 strcat(pszArgs, " ");
919 strcat(pszArgs, pszPgm);
920 strcpy(pszPgm, GetCmdSpec(FALSE)); // OS/2
921 }
922 }
923
924 // goddamned OS/2 limit
925
926 if (strlen(pszPgm) + strlen(pszArgs) > 1024)
927 pszArgs[1024 - strlen(pszPgm)] = 0;
928
929 if (!strchr(pszPgm, '\\') &&
930 !strchr(pszPgm, ':') &&
931 pszDirectory &&
932 *pszDirectory)
933 {
934 save_dir2(szSavedir);
935 switch_to(pszDirectory);
936 }
937 rc = DosQueryAppType(pszPgm,&ulAppType);
938 if (!strchr(pszPgm, '\\') &&
939 !strchr(pszPgm, ':') &&
940 pszDirectory &&
941 *pszDirectory)
942 switch_to(szSavedir);
943 if (rc) {
944 Dos_Error(MB_CANCEL,rc,hwnd,pszSrcFile,__LINE__,
945 GetPString(IDS_DOSQAPPTYPEFAILEDTEXT),
946 pszPgm, pszCallingFile, __LINE__);
947 DosFreeMem(pszPgm);
948 if (pszArgs)
949 DosFreeMem(pszArgs);
950 return -1;
951 }
952
953 if (ulAppType) {
954 if (ulAppType & (FAPPTYP_DLL | FAPPTYP_VIRTDRV | FAPPTYP_PHYSDRV | FAPPTYP_PROTDLL))
955 {
956 Runtime_Error(pszSrcFile, __LINE__,
957 GetPString(IDS_APPTYPEUNEXPECTEDTEXT),
958 pszPgm, pszCallingFile, __LINE__);
959 DosFreeMem(pszPgm);
960 if (pszArgs)
961 DosFreeMem(pszArgs);
962 return -1;
963 }
964 ulAppType &= ~FAPPTYP_BOUND;
965 if (ulAppType & (FAPPTYP_DOS | FAPPTYP_WINDOWSREAL | FAPPTYP_WINDOWSPROT | FAPPTYP_WINDOWSPROT31))
966 {
967 if (ulAppType & (FAPPTYP_WINDOWSREAL | FAPPTYP_WINDOWSPROT | FAPPTYP_WINDOWSPROT31))
968 {
969 if (~type & FULLSCREEN &&
970 ulAppType & (FAPPTYP_WINDOWSREAL | FAPPTYP_WINDOWSPROT | FAPPTYP_WINDOWSPROT31))
971 {
972 ret = RunSeamless(pszPgm, pszArgs, hwnd);
973 if (pszPgm)
974 DosFreeMem(pszPgm);
975 if (pszArgs)
976 DosFreeMem(pszArgs);
977 return ret ? 0 : -1;
978 }
979 else {
980 strcat(pszPgm, " ");
981 strcat(pszPgm, pszArgs);
982 *pszArgs = 0;
983 if (ulAppType & (FAPPTYP_WINDOWSPROT | FAPPTYP_WINDOWSREAL | FAPPTYP_WINDOWSPROT31))
984 strcat(pszArgs, "/3 ");
985 strcat(pszArgs, pszPgm);
986 strcpy(pszPgm, "WINOS2.COM");
987 }
988 }
989 else {
990 if (~type & FULLSCREEN) {
991 type |= WINDOWED;
992 ulAppType = SSF_TYPE_WINDOWEDVDM;
993 }
994 else {
995 type &= ~WINDOWED;
996 ulAppType = SSF_TYPE_VDM;
997 }
998 }
999 }
1000 else if (ulAppType & FAPPTYP_32BIT) {
1001 ulAppType &= ~FAPPTYP_32BIT;
1002 if (ulAppType == FAPPTYP_WINDOWAPI)
1003 ulAppType = SSF_TYPE_PM;
1004 else if (ulAppType == FAPPTYP_WINDOWCOMPAT)
1005 ulAppType = SSF_TYPE_WINDOWABLEVIO;
1006 else if (ulAppType == FAPPTYP_NOTWINDOWCOMPAT) {
1007 ulAppType = SSF_TYPE_FULLSCREEN;
1008 type &= ~WINDOWED;
1009 type |= FULLSCREEN;
1010 }
1011 else /* ? */
1012 ulAppType = SSF_TYPE_WINDOWABLEVIO;
1013 }
1014 else if (ulAppType == FAPPTYP_WINDOWAPI)
1015 ulAppType = SSF_TYPE_PM;
1016 else if (ulAppType == FAPPTYP_WINDOWCOMPAT)
1017 ulAppType = SSF_TYPE_WINDOWABLEVIO;
1018 else if (ulAppType == FAPPTYP_NOTWINDOWCOMPAT) {
1019 type &= ~WINDOWED;
1020 ulAppType = SSF_TYPE_FULLSCREEN;
1021 }
1022 else
1023 ulAppType = SSF_TYPE_DEFAULT;
1024 if ((type & FULLSCREEN || ~type & WINDOWED) &&
1025 ulAppType == SSF_TYPE_WINDOWABLEVIO)
1026 {
1027 ulAppType = SSF_TYPE_FULLSCREEN;
1028 }
1029 // fixme parens?
1030 else if (type & FULLSCREEN ||
1031 (type & WINDOWED && ulAppType == SSF_TYPE_WINDOWEDVDM))
1032 {
1033 ulAppType = SSF_TYPE_VDM;
1034 }
1035 }
1036 if (ulAppType == SSF_TYPE_WINDOWEDVDM && type & SEPARATEKEEP) {
1037 type &= ~SEPARATEKEEP;
1038 type |= SEPARATE;
1039 }
1040
1041 DosGetInfoBlocks(&ptib, &ppib);
1042
1043 if (~type & WAIT)
1044 useTermQ = FALSE;
1045 else {
1046 rc = 0;
1047 DosEnterCritSec();
1048 if (!hTermQ) {
1049 // Create term queue and event semaphore just once
1050 sprintf(szTermQName, TERMQ_BASE_NAME "_%x", ppib->pib_ulpid);
1051 rc = DosCreateQueue(&hTermQ, QUE_FIFO | QUE_CONVERT_ADDRESS, szTermQName);
1052 if (rc) {
1053 hTermQ = (HQUEUE)0; // Try to survive
1054 DosExitCritSec();
1055 Dos_Error(MB_CANCEL,rc,hwnd,pszSrcFile,__LINE__,"DosCreateQueue");
1056 }
1057 else {
1058 rc = DosCreateEventSem(NULL,(PHEV)&hTermQSem,0,FALSE);
1059 if (rc) {
1060 hTermQSem = (HEV)0; // Try to survive
1061 DosCloseQueue(hTermQ);
1062 hTermQ = (HQUEUE)0; // Try to survive
1063 DosExitCritSec();
1064 Dos_Error(MB_ENTER,rc,HWND_DESKTOP,pszSrcFile,__LINE__,"DoCreateEventSem");
1065 }
1066 // if (!rc) fprintf(stderr,"%s %d qcreated ptib %x hTermQ %x\n",__FILE__, __LINE__,ptib,hTermQ);
1067 }
1068 } // if 1st time
1069 useTermQ = hTermQ && hTermQSem;
1070 if (!rc)
1071 DosExitCritSec();
1072 } // if wait
1073
1074 memset(&sdata,0,sizeof(sdata));
1075 sdata.Length = sizeof(sdata);
1076 sdata.Related = type & (WAIT | CHILD) ? SSF_RELATED_CHILD :
1077 SSF_RELATED_INDEPENDENT;
1078 sdata.FgBg = type & BACKGROUND ? SSF_FGBG_BACK : SSF_FGBG_FORE;
1079 sdata.TraceOpt = SSF_TRACEOPT_NONE;
1080 sdata.PgmName = pszPgm;
1081 if (*pszArgs)
1082 sdata.PgmInputs = pszArgs;
1083 if (useTermQ)
1084 sdata.TermQ = szTermQName;
1085 sdata.Environment = pszEnvironment;
1086 sdata.InheritOpt = SSF_INHERTOPT_PARENT;
1087 sdata.SessionType = ulAppType;
1088 sdata.ObjectBuffer = szObject;
1089 sdata.ObjectBuffLen = sizeof(szObject);
1090 if ((type & RUNTYPE_MASK) == SEPARATEKEEP)
1091 sdata.PgmControl |= SSF_CONTROL_NOAUTOCLOSE;
1092 if (type & MAXIMIZED)
1093 sdata.PgmControl |= SSF_CONTROL_MAXIMIZE;
1094 if (type & MINIMIZED)
1095 sdata.PgmControl |= SSF_CONTROL_MINIMIZE;
1096 if (type & INVISIBLE)
1097 sdata.PgmControl |= SSF_CONTROL_INVISIBLE;
1098
1099 if (pszDirectory && *pszDirectory) {
1100 save_dir2(szSavedir);
1101 switch_to(pszDirectory);
1102 }
1103
1104 // printf("%s %d DosStartsession thread 0x%x data\n ",
1105 // __FILE__, __LINE__,ptib->tib_ordinal); fflush(stdout); // 10 Mar 07 SHL hang
1106 // printf(" %d %d %d %s %s %s %d %d\n %s %x %x\n",
1107 // sdata.Length , sdata.Related, sdata.FgBg, sdata.PgmName,
1108 // sdata.PgmInputs, sdata.TermQ, sdata.InheritOpt,
1109 // sdata.SessionType, szTermQName,
1110 // hTermQ, hTermQSem); fflush(stdout);
1111 ret = DosStartSession(&sdata, &ulSessID, &sessPID);
1112
1113 // if (type & WAIT) {
1114 // printf("%s %d DosStartession thread 0x%x rc = %d sess = %u pid = 0x%x\n",
1115 // __FILE__, __LINE__, ptib->tib_ordinal,ret, ulSessID, sessPID); fflush(stdout); // 10 Mar 07 SHL hang
1116 // }
1117 // else {
1118 // printf("%s %d DosStartession thread 0x%x nowait rc = %d\n",
1119 // __FILE__, __LINE__, ptib->tib_ordinal,ret); fflush(stdout); // 10 Mar 07 SHL hang
1120 // }
1121
1122 if (pszDirectory && *pszDirectory)
1123 switch_to(szSavedir);
1124
1125 if (ret && ret != ERROR_SMG_START_IN_BACKGROUND) {
1126 Dos_Error(MB_CANCEL,ret,hwnd,pszSrcFile,__LINE__,
1127 GetPString(IDS_DOSSTARTSESSIONFAILEDTEXT),pszPgm,pszArgs,
1128 pszCallingFile, __LINE__);
1129 }
1130 else if (type & WAIT) {
1131 if (!(type & (BACKGROUND | MINIMIZED | INVISIBLE)))
1132 ShowSession(hwnd, sessPID);
1133
1134 if (!useTermQ) {
1135 STATUSDATA sd;
1136 // Could not create queue - fallback - fixme to be gone?
1137 // printf("%s %d waiting wo/termq\n", __FILE__, __LINE__); fflush(stdout); // 12 Mar 07 SHL hang
1138
1139 memset(&sd, 0, sizeof(sd));
1140 sd.Length = (USHORT) sizeof(sd);
1141 sd.SelectInd = SET_SESSION_UNCHANGED;
1142 sd.BondInd = SET_SESSION_UNCHANGED;
1143 for (ctr = 0;; ctr++)
1144 {
1145 DosSleep(100);//05 Aug 07 GKY 200
1146 if (DosSetSession(ulSessID, &sd)) // Check if session gone (i.e. finished)
1147 break;
1148 if (ctr > 10) {
1149 // printf("%s %d thread 0x%x showing slow sess %u pid 0x%x\n",
1150 // __FILE__, __LINE__,ptib->tib_ordinal,ulSessID,sessPID); fflush(stdout); // 12 Mar 07 SHL
1151 ShowSession(hwnd, sessPID); // Show every 2 seconds
1152 ctr = 0;
1153 }
1154 }
1155 }
1156 else {
1157 for (ctr = 0;; ctr++)
1158 {
1159 if (ctr < 20) {
1160 rc = DosReadQueue(hTermQ, &rq, &ulLength, (PPVOID)&pTermInfo, 0,
1161 DCWW_NOWAIT, &bPriority, hTermQSem);
1162 if (rc == ERROR_QUE_EMPTY) {
1163 DosSleep(50);//05 Aug 07 GKY 100
1164 continue;
1165 }
1166 }
1167 else {
1168 if (ctr == 20) {
1169 // printf("%s %d thread 0x%x showing slow sess %u pid 0x%x\n",
1170 // __FILE__, __LINE__,ptib->tib_ordinal,ulSessID,sessPID); fflush(stdout);
1171 ShowSession(hwnd, sessPID); // Show long running session
1172 }
1173 rc = DosReadQueue(hTermQ, &rq, &ulLength, (PPVOID)&pTermInfo, 0,
1174 DCWW_WAIT, &bPriority, 0);
1175 }
1176
1177 if (rc) {
1178 // Oh heck
1179 Dos_Error(MB_CANCEL,rc,hwnd,pszSrcFile,__LINE__,"DosReadQueue");
1180 DosSleep(100);//05 Aug 07 GKY 500
1181 continue;
1182 }
1183
1184 // printf("%s %d DosReadQueue thread 0x%x sess %u sessRC %u rq.pid 0x%x rq.data 0x%x\n",
1185 // __FILE__, __LINE__,ptib->tib_ordinal,pTermInfo->usSessID,pTermInfo->usRC,rq.pid, rq.ulData); fflush(stdout);
1186
1187 if (pTermInfo->usSessID == ulSessID)
1188 break; // Our session is done
1189
1190 // Requeue session for other thread
1191 {
1192 static ULONG ulLastSessID;
1193 // printf("%s %d requeue thread 0x%x our sess %u term sess %u term rc %u\n",
1194 // __FILE__, __LINE__,ptib->tib_ordinal,ulSessID,pTermInfo->usSessID,pTermInfo->usRC); fflush(stdout);
1195 // fixme to be gone when no longer needed for debug?
1196 if (ulLastSessID) {
1197 DosSleep(100);//05 Aug 07 GKY 500
1198 ulLastSessID = pTermInfo->usSessID;
1199 }
1200 // requeue term report for other thread and do not free yet
1201 rc = DosWriteQueue(hTermQ, rq.ulData, ulLength,(PVOID)pTermInfo, bPriority);
1202 if (rc)
1203 Dos_Error(MB_CANCEL,rc,hwnd,pszSrcFile,__LINE__,"DosWriteQueue");
1204 DosSleep(50); //05 Aug 07 GKY 100 // Let other thread see queue entry
1205 }
1206 } // for
1207
1208 ret = pTermInfo->usRC == 0; // Set 1 if rc 0 else 0
1209 // printf("%s %d thread 0x%x term for sess %u\n",
1210 // __FILE__, __LINE__,ptib->tib_ordinal,ulSessID);fflush(stdout);
1211 DosFreeMem(pTermInfo);
1212 }
1213 } // if wait
1214 else if (!(type & (BACKGROUND | MINIMIZED | INVISIBLE)))
1215 ShowSession(hwnd, sessPID);
1216 }
1217 }
1218
1219ObjectInterrupt:
1220
1221 if (pszPgm)
1222 DosFreeMem(pszPgm);
1223 if (pszArgs)
1224 DosFreeMem(pszArgs);
1225
1226 return ret;
1227}
1228
1229//== Exec() Start application with WinStartApp ==
1230
1231HAPP Exec(HWND hwndNotify, BOOL child, char *startdir, char *env,
1232 PROGTYPE *progt, ULONG fl, char *formatstring,...)
1233{
1234 PROGDETAILS pgd;
1235 register char *p;
1236 char *parameters = NULL, *executable = NULL;
1237 HAPP happ = (HAPP)0;
1238 ULONG ulOptions = SAF_INSTALLEDCMDLINE;
1239 BOOL wasquote;
1240 va_list parguments;
1241
1242 if (child)
1243 ulOptions |= SAF_STARTCHILDAPP;
1244
1245 executable = xmallocz(MaxComLineStrg, pszSrcFile, __LINE__);
1246 if (executable) {
1247 va_start(parguments, formatstring);
1248 vsprintf(executable, formatstring, parguments);
1249 va_end(parguments);
1250 strip_lead_char(" \t", executable);
1251 if (*executable) {
1252 parameters = xmalloc(MaxComLineStrg, pszSrcFile, __LINE__);
1253 if (parameters) {
1254 p = executable;
1255 wasquote = FALSE;
1256 while (*p && (wasquote || (*p != ' ' && *p != '\t'))) {
1257 if (*p == '\"') {
1258 if (!wasquote) {
1259 wasquote = TRUE;
1260 memmove(p, p + 1, strlen(p));
1261 while (*p == ' ' || *p == '\t')
1262 p++;
1263 }
1264 else {
1265 memmove(p, p + 1, strlen(p));
1266 break;
1267 }
1268 }
1269 else
1270 p++;
1271 }
1272 if (*p) {
1273 *p = 0;
1274 p++;
1275 }
1276 else
1277 p = NullStr;
1278 if (*p)
1279 strcpy(parameters, p);
1280
1281 if (p && (!stricmp(p, ".BAT") || !stricmp(p, ".CMD"))) {
1282 char *temp;
1283
1284 temp = xmalloc(CCHMAXPATH * 2,pszSrcFile,__LINE__);
1285 if (temp) {
1286 if (!stricmp(p, ".BAT")) {
1287 strcpy(temp, executable);
1288 strcpy(executable, parameters);
1289 strcpy(parameters, "/C ");
1290 strcat(parameters, temp);
1291 strcat(parameters, " ");
1292 strcat(parameters, executable);
1293 strcpy(executable, GetCmdSpec(TRUE));
1294 }
1295 else if (!stricmp(p, ".CMD")) {
1296 strcpy(temp, executable);
1297 strcpy(executable, parameters);
1298 strcpy(parameters, "/C ");
1299 strcat(parameters, temp);
1300 strcat(parameters, " ");
1301 strcat(parameters, executable);
1302 strcpy(executable, GetCmdSpec(FALSE));
1303 }
1304 xfree(temp);
1305 }
1306 }
1307
1308 memset(&pgd, 0, sizeof(pgd));
1309 pgd.Length = sizeof(pgd);
1310 pgd.progt = *progt;
1311 pgd.swpInitial.fl = fl;
1312 pgd.pszEnvironment = env;
1313 pgd.pszStartupDir = startdir;
1314 pgd.pszParameters = *parameters ? parameters : NULL;
1315 pgd.pszExecutable = executable;
1316 pgd.swpInitial.hwndInsertBehind = HWND_TOP;
1317 happ = WinStartApp(hwndNotify, &pgd, NULL, NULL, ulOptions);
1318 xfree(parameters);
1319 }
1320 }
1321 xfree(executable);
1322 }
1323 return happ;
1324}
1325
1326#pragma alloc_text(SYSTEMF,ShowSession,ExecOnList,runemf2)
Note: See TracBrowser for help on using the repository browser.