2 * Wine virtual DOS machine
4 * Copyright 2003 Alexandre Julliard
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22 #include "wine/port.h"
30 #include "wine/winbase16.h"
33 #include "wine/unicode.h"
34 #include "wine/debug.h"
36 WINE_DEFAULT_DEBUG_CHANNEL(winevdm
);
38 extern void __wine_load_dos_exe( LPCSTR filename
, LPCSTR cmdline
);
41 /*** PIF file structures ***/
44 /* header of a PIF file */
46 BYTE unk1
[2]; /* 0x00 */
47 CHAR windowtitle
[ 30 ]; /* 0x02 seems to be padded with blanks*/
48 WORD memmax
; /* 0x20 */
49 WORD memmin
; /* 0x22 */
50 CHAR program
[63]; /* 0x24 seems to be zero terminated */
51 BYTE hdrflags1
; /* 0x63 various flags:
52 * 02 286: text mode selected
53 * 10 close window at exit
55 BYTE startdrive
; /* 0x64 */
56 char startdir
[64]; /* 0x65 */
57 char optparams
[64]; /* 0xa5 seems to be zero terminated */
58 BYTE videomode
; /* 0xe5 */
59 BYTE unkn2
; /* 0xe6 ?*/
60 BYTE irqlow
; /* 0xe7 */
61 BYTE irqhigh
; /* 0xe8 */
66 WORD unkn3
; /* 0xed 7??? */
67 CHAR unkn4
[64]; /* 0xef */
68 CHAR unkn5
[64]; /* 0x12f */
69 BYTE hdrflags2
; /* 0x16f */
70 BYTE hdrflags3
; /* 0x170 */
73 /* record header: present on every record */
75 CHAR recordname
[16]; /* zero terminated */
76 WORD posofnextrecord
; /* file offset, 0xffff if last */
77 WORD startofdata
; /* file offset */
78 WORD sizeofdata
; /* data is expected to follow directly */
81 /* 386 -enhanced mode- record */
83 WORD memmax
; /* memory desired, overrides the pif header*/
84 WORD memmin
; /* memory required, overrides the pif header*/
85 WORD prifg
; /* foreground priority */
86 WORD pribg
; /* background priority */
87 WORD emsmax
; /* EMS memory limit */
88 WORD emsmin
; /* EMS memory required */
89 WORD xmsmax
; /* XMS memory limit */
90 WORD xmsmin
; /* XMS memory required */
91 WORD optflags
; /* option flags:
95 * 0001 close when active
97 WORD memflags
; /* various memory flags*/
98 WORD videoflags
; /* video flags:
100 * 0020 med. res. graphics
101 * 0040 hi. res. graphics
103 WORD hotkey
[9]; /* Hot key info */
104 CHAR optparams
[64]; /* optional params, replaces those in the pif header */
109 /***********************************************************************
112 static char *find_dosbox(void)
114 const char *envpath
= getenv( "PATH" );
116 char *path
, *p
, *buffer
, *dir
;
119 if (!envpath
) return NULL
;
121 envpath_len
= strlen( envpath
);
122 path
= HeapAlloc( GetProcessHeap(), 0, envpath_len
+ 1 );
123 buffer
= HeapAlloc( GetProcessHeap(), 0, envpath_len
+ sizeof("/dosbox") );
124 strcpy( path
, envpath
);
129 while (*p
== ':') p
++;
132 while (*p
&& *p
!= ':') p
++;
133 if (*p
== ':') *p
++ = 0;
134 strcpy( buffer
, dir
);
135 strcat( buffer
, "/dosbox" );
136 if (!stat( buffer
, &st
))
138 HeapFree( GetProcessHeap(), 0, path
);
142 HeapFree( GetProcessHeap(), 0, buffer
);
143 HeapFree( GetProcessHeap(), 0, path
);
148 /***********************************************************************
151 static void start_dosbox( const char *appname
, const char *args
)
153 static const WCHAR cfgW
[] = {'c','f','g',0};
154 const char *config_dir
= wine_get_config_dir();
155 WCHAR path
[MAX_PATH
], config
[MAX_PATH
];
157 char *p
, *buffer
, app
[MAX_PATH
];
160 DWORD written
, drives
= GetLogicalDrives();
161 char *dosbox
= find_dosbox();
164 if (!GetTempPathW( MAX_PATH
, path
)) return;
165 if (!GetTempFileNameW( path
, cfgW
, 0, config
)) return;
166 if (!GetCurrentDirectoryW( MAX_PATH
, path
)) return;
167 if (!GetShortPathNameA( appname
, app
, MAX_PATH
)) return;
168 GetShortPathNameW( path
, path
, MAX_PATH
);
169 file
= CreateFileW( config
, GENERIC_WRITE
, 0, NULL
, CREATE_ALWAYS
, 0, 0 );
170 if (file
== INVALID_HANDLE_VALUE
) return;
172 buffer
= HeapAlloc( GetProcessHeap(), 0, sizeof("[autoexec]") +
173 sizeof("mount -z c") + sizeof("config -securemode") +
174 25 * (strlen(config_dir
) + sizeof("mount c /dosdevices/c:")) +
175 4 * strlenW( path
) +
176 6 + strlen( app
) + strlen( args
) + 20 );
178 p
+= sprintf( p
, "[autoexec]\n" );
179 for (i
= 25; i
>= 0; i
--)
180 if (!(drives
& (1 << i
)))
182 p
+= sprintf( p
, "mount -z %c\n", 'a' + i
);
185 for (i
= 0; i
<= 25; i
++)
186 if (drives
& (1 << i
))
187 p
+= sprintf( p
, "mount %c %s/dosdevices/%c:\n", 'a' + i
, config_dir
, 'a' + i
);
188 p
+= sprintf( p
, "%c:\ncd ", path
[0] );
189 p
+= WideCharToMultiByte( CP_UNIXCP
, 0, path
+ 2, -1, p
, 4 * strlenW(path
), NULL
, NULL
) - 1;
190 p
+= sprintf( p
, "\nconfig -securemode\n" );
191 p
+= sprintf( p
, "%s %s\n", app
, args
);
192 p
+= sprintf( p
, "exit\n" );
193 if (WriteFile( file
, buffer
, strlen(buffer
), &written
, NULL
) && written
== strlen(buffer
))
196 char *config_file
= wine_get_unix_file_name( config
);
199 args
[2] = config_file
;
201 ret
= _spawnvp( _P_WAIT
, args
[0], args
);
204 DeleteFileW( config
);
205 HeapFree( GetProcessHeap(), 0, buffer
);
210 /***********************************************************************
213 static void start_dos_exe( LPCSTR filename
, LPCSTR cmdline
)
215 MEMORY_BASIC_INFORMATION mem_info
;
218 start_dosbox( filename
, cmdline
);
220 if (VirtualQuery( NULL
, &mem_info
, sizeof(mem_info
) ) && mem_info
.State
!= MEM_FREE
)
222 __wine_load_dos_exe( filename
, cmdline
);
223 if (GetLastError() == ERROR_NOT_SUPPORTED
)
224 reason
= "because vm86 mode is not supported on this platform";
226 reason
= wine_dbg_sprintf( "It failed with error code %u", GetLastError() );
228 else reason
= "because the DOS memory range is unavailable";
230 WINE_MESSAGE( "winevdm: Cannot start DOS application %s\n", filename
);
231 WINE_MESSAGE( " %s.\n", reason
);
232 WINE_MESSAGE( " You should install DOSBox.\n" );
236 /***********************************************************************
239 * Read a pif file and return the header and possibly the 286 (real mode)
240 * record or 386 (enhanced mode) record. Returns FALSE if the file is
241 * invalid otherwise TRUE.
243 static BOOL
read_pif_file( HANDLE hFile
, char *progname
, char *title
,
244 char *optparams
, char *startdir
, int *closeonexit
, int *textmode
)
247 LARGE_INTEGER filesize
;
249 BOOL found386rec
= FALSE
;
250 pif386rec_t pif386rec
;
252 if( !GetFileSizeEx( hFile
, &filesize
) ||
253 filesize
.QuadPart
< (sizeof(pifhead_t
) + sizeof(recordhead_t
))) {
254 WINE_ERR("Invalid pif file: size error %d\n", (int)filesize
.QuadPart
);
257 SetFilePointer( hFile
, 0, NULL
, FILE_BEGIN
);
258 if( !ReadFile( hFile
, &pifheader
, sizeof(pifhead_t
), &nread
, NULL
))
260 WINE_TRACE("header: program %s title %s startdir %s params %s\n",
261 wine_dbgstr_a(pifheader
.program
),
262 wine_dbgstr_an(pifheader
.windowtitle
, sizeof(pifheader
.windowtitle
)),
263 wine_dbgstr_a(pifheader
.startdir
),
264 wine_dbgstr_a(pifheader
.optparams
));
265 WINE_TRACE("header: memory req'd %d desr'd %d drive %d videomode %d\n",
266 pifheader
.memmin
, pifheader
.memmax
, pifheader
.startdrive
,
267 pifheader
.videomode
);
268 WINE_TRACE("header: flags 0x%x 0x%x 0x%x\n",
269 pifheader
.hdrflags1
, pifheader
.hdrflags2
, pifheader
.hdrflags3
);
270 ReadFile( hFile
, &rhead
, sizeof(recordhead_t
), &nread
, NULL
);
271 if( strncmp( rhead
.recordname
, "MICROSOFT PIFEX", 15)) {
272 WINE_ERR("Invalid pif file: magic string not found\n");
275 /* now process the following records */
277 WORD nextrecord
= rhead
.posofnextrecord
;
278 if( (nextrecord
& 0x8000) ||
279 filesize
.QuadPart
<( nextrecord
+ sizeof(recordhead_t
))) break;
280 if( !SetFilePointer( hFile
, nextrecord
, NULL
, FILE_BEGIN
) ||
281 !ReadFile( hFile
, &rhead
, sizeof(recordhead_t
), &nread
, NULL
))
283 if( !rhead
.recordname
[0]) continue; /* deleted record */
284 WINE_TRACE("reading record %s size %d next 0x%x\n",
285 wine_dbgstr_a(rhead
.recordname
), rhead
.sizeofdata
,
286 rhead
.posofnextrecord
);
287 if( !strncmp( rhead
.recordname
, "WINDOWS 386", 11)) {
289 ReadFile( hFile
, &pif386rec
, sizeof(pif386rec_t
), &nread
, NULL
);
290 WINE_TRACE("386rec: memory req'd %d des'd %d EMS req'd %d des'd %d XMS req'd %d des'd %d\n",
291 pif386rec
.memmin
, pif386rec
.memmax
,
292 pif386rec
.emsmin
, pif386rec
.emsmax
,
293 pif386rec
.xmsmin
, pif386rec
.xmsmax
);
294 WINE_TRACE("386rec: option 0x%x memory 0x%x video 0x%x\n",
295 pif386rec
.optflags
, pif386rec
.memflags
,
296 pif386rec
.videoflags
);
297 WINE_TRACE("386rec: optional parameters %s\n",
298 wine_dbgstr_a(pif386rec
.optparams
));
301 /* prepare the return data */
302 lstrcpynA( progname
, pifheader
.program
, sizeof(pifheader
.program
)+1);
303 lstrcpynA( title
, pifheader
.windowtitle
, sizeof(pifheader
.windowtitle
)+1);
305 lstrcpynA( optparams
, pif386rec
.optparams
, sizeof( pif386rec
.optparams
)+1);
307 lstrcpynA( optparams
, pifheader
.optparams
, sizeof(pifheader
.optparams
)+1);
308 lstrcpynA( startdir
, pifheader
.startdir
, sizeof(pifheader
.startdir
)+1);
309 *closeonexit
= pifheader
.hdrflags1
& 0x10;
310 *textmode
= found386rec
? pif386rec
.videoflags
& 0x0010
311 : pifheader
.hdrflags1
& 0x0002;
315 /***********************************************************************
318 * execute a pif file.
320 static VOID
pif_cmd( char *filename
, char *cmdline
)
323 char progpath
[MAX_PATH
];
332 if( (hFile
= CreateFileA( filename
, GENERIC_READ
, FILE_SHARE_READ
,
333 NULL
, OPEN_EXISTING
, 0, 0 )) == INVALID_HANDLE_VALUE
)
335 WINE_ERR("open file %s failed\n", wine_dbgstr_a(filename
));
338 if( !read_pif_file( hFile
, progname
, title
, optparams
, startdir
,
339 &closeonexit
, &textmode
)) {
340 WINE_ERR( "failed to read %s\n", wine_dbgstr_a(filename
));
342 sprintf( buf
, "%s\nInvalid file format. Check your pif file.",
344 MessageBoxA( NULL
, buf
, "16 bit DOS subsystem", MB_OK
|MB_ICONWARNING
);
345 SetLastError( ERROR_BAD_FORMAT
);
349 if( (p
= strrchr( progname
, '.')) && !strcasecmp( p
, ".bat"))
350 WINE_FIXME(".bat programs in pif files are not supported.\n");
351 /* first change dir, so the search below can start from there */
352 if( startdir
[0] && !SetCurrentDirectoryA( startdir
)) {
353 WINE_ERR("Cannot change directory %s\n", wine_dbgstr_a( startdir
));
354 sprintf( buf
, "%s\nInvalid startup directory. Check your pif file.",
356 MessageBoxA( NULL
, buf
, "16 bit DOS subsystem", MB_OK
|MB_ICONWARNING
);
358 /* search for the program */
359 if( !SearchPathA( NULL
, progname
, NULL
, MAX_PATH
, progpath
, NULL
)) {
360 sprintf( buf
, "%s\nInvalid program file name. Check your pif file.",
362 MessageBoxA( NULL
, buf
, "16 bit DOS subsystem", MB_OK
|MB_ICONERROR
);
363 SetLastError( ERROR_FILE_NOT_FOUND
);
368 SetConsoleTitleA( title
) ;
369 /* if no arguments on the commandline, use them from the pif file */
370 if( !cmdline
[0] && optparams
[0])
372 /* FIXME: do something with:
378 start_dos_exe( progpath
, cmdline
);
381 /***********************************************************************
384 * Build the command line of a process from the argv array.
385 * Copied from ENV_BuildCommandLine.
387 static char *build_command_line( char **argv
)
390 char *p
, **arg
, *cmd_line
;
393 for (arg
= argv
; *arg
; arg
++)
402 if( !*a
) has_space
=TRUE
;
407 if (*a
==' ' || *a
=='\t') {
409 } else if (*a
=='"') {
410 /* doubling of '\' preceding a '"',
411 * plus escaping of said '"'
419 len
+=(a
-*arg
)+1 /* for the separating space */;
421 len
+=2; /* for the quotes */
424 if (!(cmd_line
= HeapAlloc( GetProcessHeap(), 0, len
? len
+ 1 : 2 )))
428 *p
++ = (len
< 256) ? len
: 255;
429 for (arg
= argv
; *arg
; arg
++)
431 BOOL has_space
,has_quote
;
434 /* Check for quotes and spaces in this argument */
435 has_space
=has_quote
=FALSE
;
437 if( !*a
) has_space
=TRUE
;
439 if (*a
==' ' || *a
=='\t') {
443 } else if (*a
=='"') {
451 /* Now transfer it to the command line */
467 /* Double all the '\\' preceding this '"', plus one */
468 for (i
=0;i
<=bcount
;i
++)
486 if (len
) p
--; /* remove last space */
492 /***********************************************************************
495 static void usage(void)
497 WINE_MESSAGE( "Usage: winevdm.exe [--app-name app.exe] command line\n\n" );
502 /***********************************************************************
505 int main( int argc
, char *argv
[] )
508 HINSTANCE16 instance
;
511 char buffer
[MAX_PATH
];
513 char *cmdline
, *appname
, **first_arg
;
516 if (!argv
[1]) usage();
518 if (!strcmp( argv
[1], "--app-name" ))
520 if (!(appname
= argv
[2])) usage();
521 first_arg
= argv
+ 3;
525 if (!SearchPathA( NULL
, argv
[1], ".exe", sizeof(buffer
), buffer
, NULL
))
527 WINE_MESSAGE( "winevdm: unable to exec '%s': file not found\n", argv
[1] );
531 first_arg
= argv
+ 1;
534 if (*first_arg
) first_arg
++; /* skip program name */
535 cmdline
= build_command_line( first_arg
);
537 if (WINE_TRACE_ON(winevdm
))
540 WINE_TRACE( "GetCommandLine = '%s'\n", GetCommandLineA() );
541 WINE_TRACE( "appname = '%s'\n", appname
);
542 WINE_TRACE( "cmdline = '%.*s'\n", cmdline
[0], cmdline
+1 );
543 for (i
= 0; argv
[i
]; i
++) WINE_TRACE( "argv[%d]: '%s'\n", i
, argv
[i
] );
546 GetStartupInfoA( &info
);
548 showCmd
[1] = (info
.dwFlags
& STARTF_USESHOWWINDOW
) ? info
.wShowWindow
: SW_SHOWNORMAL
;
550 params
.hEnvironment
= 0;
551 params
.cmdLine
= MapLS( cmdline
);
552 params
.showCmd
= MapLS( showCmd
);
555 RestoreThunkLock(1); /* grab the Win16 lock */
557 /* some programs assume mmsystem is always present */
558 LoadLibrary16( "gdi.exe" );
559 LoadLibrary16( "user.exe" );
560 LoadLibrary16( "mmsystem.dll" );
562 if ((instance
= LoadModule16( appname
, ¶ms
)) < 32)
566 /* first see if it is a .pif file */
567 if( ( p
= strrchr( appname
, '.' )) && !strcasecmp( p
, ".pif"))
568 pif_cmd( appname
, cmdline
+ 1);
572 /* loader expects arguments to be regular C strings */
573 start_dos_exe( appname
, cmdline
+ 1 );
575 /* if we get back here it failed */
576 instance
= GetLastError();
579 WINE_MESSAGE( "winevdm: can't exec '%s': ", appname
);
582 case 2: WINE_MESSAGE("file not found\n" ); break;
583 case 11: WINE_MESSAGE("invalid program file\n" ); break;
584 default: WINE_MESSAGE("error=%d\n", instance
); break;
586 ExitProcess(instance
);
589 /* wait forever; the process will be killed when the last task exits */
590 ReleaseThunkLock( &count
);