|  | /* | 
|  | * Wine virtual DOS machine | 
|  | * | 
|  | * Copyright 2003 Alexandre Julliard | 
|  | * | 
|  | * This library is free software; you can redistribute it and/or | 
|  | * modify it under the terms of the GNU Lesser General Public | 
|  | * License as published by the Free Software Foundation; either | 
|  | * version 2.1 of the License, or (at your option) any later version. | 
|  | * | 
|  | * This library is distributed in the hope that it will be useful, | 
|  | * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
|  | * Lesser General Public License for more details. | 
|  | * | 
|  | * You should have received a copy of the GNU Lesser General Public | 
|  | * License along with this library; if not, write to the Free Software | 
|  | * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA | 
|  | */ | 
|  |  | 
|  | #include <stdarg.h> | 
|  | #include <stdio.h> | 
|  |  | 
|  | #include "windef.h" | 
|  | #include "winbase.h" | 
|  | #include "wine/winbase16.h" | 
|  | #include "winuser.h" | 
|  | #include "wincon.h" | 
|  | #include "wine/debug.h" | 
|  |  | 
|  | WINE_DEFAULT_DEBUG_CHANNEL(winevdm); | 
|  |  | 
|  | static void (WINAPI *wine_load_dos_exe)( LPCSTR filename, LPCSTR cmdline ); | 
|  |  | 
|  |  | 
|  | /*** PIF file structures ***/ | 
|  | #include "pshpack1.h" | 
|  |  | 
|  | /* header of a PIF file */ | 
|  | typedef struct { | 
|  | BYTE unk1[2];               /* 0x00 */ | 
|  | CHAR windowtitle[ 30 ];     /* 0x02 seems to be padded with blanks*/ | 
|  | WORD memmax;                /* 0x20 */ | 
|  | WORD memmin;                /* 0x22 */ | 
|  | CHAR program[63];           /* 0x24 seems to be zero terminated */ | 
|  | BYTE hdrflags1;             /* 0x63 various flags: | 
|  | *  02 286: text mode selected | 
|  | *  10 close window at exit | 
|  | */ | 
|  | BYTE startdrive;            /* 0x64 */ | 
|  | char startdir[64];          /* 0x65 */ | 
|  | char optparams[64];         /* 0xa5 seems to be zero terminated */ | 
|  | BYTE videomode;             /* 0xe5 */ | 
|  | BYTE unkn2;                 /* 0xe6 ?*/ | 
|  | BYTE irqlow;                /* 0xe7 */ | 
|  | BYTE irqhigh;               /* 0xe8 */ | 
|  | BYTE rows;                  /* 0xe9 */ | 
|  | BYTE cols;                  /* 0xea */ | 
|  | BYTE winY;                  /* 0xeb */ | 
|  | BYTE winX;                  /* 0xec */ | 
|  | WORD unkn3;                 /* 0xed 7??? */ | 
|  | CHAR unkn4[64];             /* 0xef */ | 
|  | CHAR unkn5[64];             /* 0x12f */ | 
|  | BYTE hdrflags2;             /* 0x16f */ | 
|  | BYTE hdrflags3;             /* 0x170 */ | 
|  | } pifhead_t; | 
|  |  | 
|  | /* record header: present on every record */ | 
|  | typedef struct { | 
|  | CHAR recordname[16];  /* zero terminated */ | 
|  | WORD posofnextrecord; /* file offset, 0xffff if last */ | 
|  | WORD startofdata;     /* file offset */ | 
|  | WORD sizeofdata;      /* data is expected to follow directly */ | 
|  | } recordhead_t; | 
|  |  | 
|  | /* 386 -enhanced mode- record */ | 
|  | typedef struct { | 
|  | WORD memmax;         /* memory desired, overrides the pif header*/ | 
|  | WORD memmin;         /* memory required, overrides the pif header*/ | 
|  | WORD prifg;          /* foreground priority */ | 
|  | WORD pribg;          /* background priority */ | 
|  | WORD emsmax;         /* EMS memory limit */ | 
|  | WORD emsmin;         /* EMS memory required */ | 
|  | WORD xmsmax;         /* XMS memory limit */ | 
|  | WORD xmsmin;         /* XMS memory required */ | 
|  | WORD optflags;        /* option flags: | 
|  | *  0008 full screen | 
|  | *  0004 exclusive | 
|  | *  0002 background | 
|  | *  0001 close when active | 
|  | */ | 
|  | WORD memflags;       /* various memory flags*/ | 
|  | WORD videoflags;     /* video flags: | 
|  | *   0010 text | 
|  | *   0020 med. res. graphics | 
|  | *   0040 hi. res. graphics | 
|  | */ | 
|  | WORD hotkey[9];      /* Hot key info */ | 
|  | CHAR optparams[64];  /* optional params, replaces those in the pif header */ | 
|  | } pif386rec_t; | 
|  |  | 
|  | #include "poppack.h" | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           read_pif_file | 
|  | *pif386rec_tu | 
|  | * Read a pif file and return the header and possibly the 286 (real mode) | 
|  | * record or 386 (enhanced mode) record. Returns FALSE if the file is | 
|  | * invalid otherwise TRUE. | 
|  | */ | 
|  | static BOOL read_pif_file( HANDLE hFile, char *progname, char *title, | 
|  | char *optparams, char *startdir, int *closeonexit, int *textmode) | 
|  | { | 
|  | DWORD nread; | 
|  | LARGE_INTEGER filesize; | 
|  | recordhead_t rhead; | 
|  | BOOL found386rec = FALSE; | 
|  | pif386rec_t pif386rec; | 
|  | pifhead_t pifheader; | 
|  | if( !GetFileSizeEx( hFile, &filesize) || | 
|  | filesize.QuadPart <  (sizeof(pifhead_t) + sizeof(recordhead_t))) { | 
|  | WINE_ERR("Invalid pif file: size error %d\n", (int)filesize.QuadPart); | 
|  | return FALSE; | 
|  | } | 
|  | SetFilePointer( hFile, 0, NULL, FILE_BEGIN); | 
|  | if( !ReadFile( hFile, &pifheader, sizeof(pifhead_t), &nread, NULL)) | 
|  | return FALSE; | 
|  | WINE_TRACE("header: program %s title %s startdir %s params %s\n", | 
|  | wine_dbgstr_a(pifheader.program), | 
|  | wine_dbgstr_an(pifheader.windowtitle, sizeof(pifheader.windowtitle)), | 
|  | wine_dbgstr_a(pifheader.startdir), | 
|  | wine_dbgstr_a(pifheader.optparams)); | 
|  | WINE_TRACE("header: memory req'd %d desr'd %d drive %d videomode %d\n", | 
|  | pifheader.memmin, pifheader.memmax, pifheader.startdrive, | 
|  | pifheader.videomode); | 
|  | WINE_TRACE("header: flags 0x%x 0x%x 0x%x\n", | 
|  | pifheader.hdrflags1, pifheader.hdrflags2, pifheader.hdrflags3); | 
|  | ReadFile( hFile, &rhead, sizeof(recordhead_t), &nread, NULL); | 
|  | if( strncmp( rhead.recordname, "MICROSOFT PIFEX", 15)) { | 
|  | WINE_ERR("Invalid pif file: magic string not found\n"); | 
|  | return FALSE; | 
|  | } | 
|  | /* now process the following records */ | 
|  | while( 1) { | 
|  | WORD nextrecord = rhead.posofnextrecord; | 
|  | if( (nextrecord & 0x8000) || | 
|  | filesize.QuadPart <( nextrecord + sizeof(recordhead_t))) break; | 
|  | if( !SetFilePointer( hFile, nextrecord, NULL, FILE_BEGIN) || | 
|  | !ReadFile( hFile, &rhead, sizeof(recordhead_t), &nread, NULL)) | 
|  | return FALSE; | 
|  | if( !rhead.recordname[0]) continue; /* deleted record */ | 
|  | WINE_TRACE("reading record %s size %d next 0x%x\n", | 
|  | wine_dbgstr_a(rhead.recordname), rhead.sizeofdata, | 
|  | rhead.posofnextrecord ); | 
|  | if( !strncmp( rhead.recordname, "WINDOWS 386", 11)) { | 
|  | found386rec = TRUE; | 
|  | ReadFile( hFile, &pif386rec, sizeof(pif386rec_t), &nread, NULL); | 
|  | 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", | 
|  | pif386rec.memmin, pif386rec.memmax, | 
|  | pif386rec.emsmin, pif386rec.emsmax, | 
|  | pif386rec.xmsmin, pif386rec.xmsmax); | 
|  | WINE_TRACE("386rec: option 0x%x memory 0x%x video 0x%x\n", | 
|  | pif386rec.optflags, pif386rec.memflags, | 
|  | pif386rec.videoflags); | 
|  | WINE_TRACE("386rec: optional parameters %s\n", | 
|  | wine_dbgstr_a(pif386rec.optparams)); | 
|  | } | 
|  | } | 
|  | /* prepare the return data */ | 
|  | strncpy( progname, pifheader.program, sizeof(pifheader.program)); | 
|  | memcpy( title, pifheader.windowtitle, sizeof(pifheader.windowtitle)); | 
|  | title[ sizeof(pifheader.windowtitle) ] = '\0'; | 
|  | if( found386rec) | 
|  | strncpy( optparams, pif386rec.optparams, sizeof( pif386rec.optparams)); | 
|  | else | 
|  | strncpy( optparams, pifheader.optparams, sizeof(pifheader.optparams)); | 
|  | strncpy( startdir, pifheader.startdir, sizeof(pifheader.startdir)); | 
|  | *closeonexit = pifheader.hdrflags1 & 0x10; | 
|  | *textmode = found386rec ? pif386rec.videoflags & 0x0010 | 
|  | : pifheader.hdrflags1 & 0x0002; | 
|  | return TRUE; | 
|  | } | 
|  |  | 
|  | /*********************************************************************** | 
|  | *              pif_cmd | 
|  | * | 
|  | * execute a pif file. | 
|  | */ | 
|  | static VOID pif_cmd( char *filename, char *cmdline) | 
|  | { | 
|  | HANDLE hFile; | 
|  | char progpath[MAX_PATH]; | 
|  | char buf[128]; | 
|  | char progname[64]; | 
|  | char title[31]; | 
|  | char optparams[64]; | 
|  | char startdir[64]; | 
|  | char *p; | 
|  | int closeonexit; | 
|  | int textmode; | 
|  | if( (hFile = CreateFileA( filename, GENERIC_READ, FILE_SHARE_READ, | 
|  | NULL, OPEN_EXISTING, 0, 0 )) == INVALID_HANDLE_VALUE) | 
|  | { | 
|  | WINE_ERR("open file %s failed\n", wine_dbgstr_a(filename)); | 
|  | return; | 
|  | } | 
|  | if( !read_pif_file( hFile, progname, title, optparams, startdir, | 
|  | &closeonexit, &textmode)) { | 
|  | WINE_ERR( "failed to read %s\n", wine_dbgstr_a(filename)); | 
|  | CloseHandle( hFile); | 
|  | sprintf( buf, "%s\nInvalid file format. Check your pif file.", | 
|  | filename); | 
|  | MessageBoxA( NULL, buf, "16 bit DOS subsystem", MB_OK|MB_ICONWARNING); | 
|  | SetLastError( ERROR_BAD_FORMAT); | 
|  | return; | 
|  | } | 
|  | CloseHandle( hFile); | 
|  | if( (p = strrchr( progname, '.')) && !strcasecmp( p, ".bat")) | 
|  | WINE_FIXME(".bat programs in pif files are not supported.\n"); | 
|  | /* first change dir, so the search below can start from there */ | 
|  | if( startdir[0] && !SetCurrentDirectoryA( startdir)) { | 
|  | WINE_ERR("Cannot change directory %s\n", wine_dbgstr_a( startdir)); | 
|  | sprintf( buf, "%s\nInvalid startup directory. Check your pif file.", | 
|  | filename); | 
|  | MessageBoxA( NULL, buf, "16 bit DOS subsystem", MB_OK|MB_ICONWARNING); | 
|  | } | 
|  | /* search for the program */ | 
|  | if( !SearchPathA( NULL, progname, NULL, MAX_PATH, progpath, NULL )) { | 
|  | sprintf( buf, "%s\nInvalid program file name. Check your pif file.", | 
|  | filename); | 
|  | MessageBoxA( NULL, buf, "16 bit DOS subsystem", MB_OK|MB_ICONERROR); | 
|  | SetLastError( ERROR_FILE_NOT_FOUND); | 
|  | return; | 
|  | } | 
|  | if( textmode) | 
|  | if( AllocConsole()) | 
|  | SetConsoleTitleA( title) ; | 
|  | /* if no arguments on the commandline, use them from the pif file */ | 
|  | if( !cmdline[0] && optparams[0]) | 
|  | cmdline = optparams; | 
|  | /* FIXME: do something with: | 
|  | * - close on exit | 
|  | * - graphic modes | 
|  | * - hot key's | 
|  | * - etc. | 
|  | */ | 
|  | wine_load_dos_exe( progpath, cmdline ); | 
|  | return; | 
|  | } | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           build_command_line | 
|  | * | 
|  | * Build the command line of a process from the argv array. | 
|  | * Copied from ENV_BuildCommandLine. | 
|  | */ | 
|  | static char *build_command_line( char **argv ) | 
|  | { | 
|  | int len; | 
|  | char *p, **arg, *cmd_line; | 
|  |  | 
|  | len = 0; | 
|  | for (arg = argv; *arg; arg++) | 
|  | { | 
|  | int has_space,bcount; | 
|  | char* a; | 
|  |  | 
|  | has_space=0; | 
|  | bcount=0; | 
|  | a=*arg; | 
|  | if( !*a ) has_space=1; | 
|  | while (*a!='\0') { | 
|  | if (*a=='\\') { | 
|  | bcount++; | 
|  | } else { | 
|  | if (*a==' ' || *a=='\t') { | 
|  | has_space=1; | 
|  | } else if (*a=='"') { | 
|  | /* doubling of '\' preceding a '"', | 
|  | * plus escaping of said '"' | 
|  | */ | 
|  | len+=2*bcount+1; | 
|  | } | 
|  | bcount=0; | 
|  | } | 
|  | a++; | 
|  | } | 
|  | len+=(a-*arg)+1 /* for the separating space */; | 
|  | if (has_space) | 
|  | len+=2; /* for the quotes */ | 
|  | } | 
|  |  | 
|  | if (!(cmd_line = HeapAlloc( GetProcessHeap(), 0, len ? len + 1 : 2 ))) | 
|  | return NULL; | 
|  |  | 
|  | p = cmd_line; | 
|  | *p++ = (len < 256) ? len : 255; | 
|  | for (arg = argv; *arg; arg++) | 
|  | { | 
|  | int has_space,has_quote; | 
|  | char* a; | 
|  |  | 
|  | /* Check for quotes and spaces in this argument */ | 
|  | has_space=has_quote=0; | 
|  | a=*arg; | 
|  | if( !*a ) has_space=1; | 
|  | while (*a!='\0') { | 
|  | if (*a==' ' || *a=='\t') { | 
|  | has_space=1; | 
|  | if (has_quote) | 
|  | break; | 
|  | } else if (*a=='"') { | 
|  | has_quote=1; | 
|  | if (has_space) | 
|  | break; | 
|  | } | 
|  | a++; | 
|  | } | 
|  |  | 
|  | /* Now transfer it to the command line */ | 
|  | if (has_space) | 
|  | *p++='"'; | 
|  | if (has_quote) { | 
|  | int bcount; | 
|  | char* a; | 
|  |  | 
|  | bcount=0; | 
|  | a=*arg; | 
|  | while (*a!='\0') { | 
|  | if (*a=='\\') { | 
|  | *p++=*a; | 
|  | bcount++; | 
|  | } else { | 
|  | if (*a=='"') { | 
|  | int i; | 
|  |  | 
|  | /* Double all the '\\' preceding this '"', plus one */ | 
|  | for (i=0;i<=bcount;i++) | 
|  | *p++='\\'; | 
|  | *p++='"'; | 
|  | } else { | 
|  | *p++=*a; | 
|  | } | 
|  | bcount=0; | 
|  | } | 
|  | a++; | 
|  | } | 
|  | } else { | 
|  | strcpy(p,*arg); | 
|  | p+=strlen(*arg); | 
|  | } | 
|  | if (has_space) | 
|  | *p++='"'; | 
|  | *p++=' '; | 
|  | } | 
|  | if (len) p--;  /* remove last space */ | 
|  | *p = '\0'; | 
|  | return cmd_line; | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           usage | 
|  | */ | 
|  | static void usage(void) | 
|  | { | 
|  | WINE_MESSAGE( "Usage: winevdm.exe [--app-name app.exe] command line\n\n" ); | 
|  | ExitProcess(1); | 
|  | } | 
|  |  | 
|  |  | 
|  | /*********************************************************************** | 
|  | *           main | 
|  | */ | 
|  | int main( int argc, char *argv[] ) | 
|  | { | 
|  | DWORD count; | 
|  | HINSTANCE16 instance; | 
|  | LOADPARAMS16 params; | 
|  | WORD showCmd[2]; | 
|  | char buffer[MAX_PATH]; | 
|  | STARTUPINFOA info; | 
|  | char *cmdline, *appname, **first_arg; | 
|  | char *p; | 
|  | HMODULE winedos; | 
|  | MEMORY_BASIC_INFORMATION mem_info; | 
|  |  | 
|  | if (!argv[1]) usage(); | 
|  |  | 
|  | if (!strcmp( argv[1], "--app-name" )) | 
|  | { | 
|  | if (!(appname = argv[2])) usage(); | 
|  | first_arg = argv + 3; | 
|  | } | 
|  | else | 
|  | { | 
|  | if (!SearchPathA( NULL, argv[1], ".exe", sizeof(buffer), buffer, NULL )) | 
|  | { | 
|  | WINE_MESSAGE( "winevdm: unable to exec '%s': file not found\n", argv[1] ); | 
|  | ExitProcess(1); | 
|  | } | 
|  | appname = buffer; | 
|  | first_arg = argv + 1; | 
|  | } | 
|  |  | 
|  | if (!(winedos = LoadLibraryA( "winedos.dll" )) || | 
|  | !(wine_load_dos_exe = (void *)GetProcAddress( winedos, "wine_load_dos_exe" ))) | 
|  | { | 
|  | WINE_MESSAGE( "winevdm: unable to exec '%s': DOS support unavailable\n", appname ); | 
|  | ExitProcess(1); | 
|  | } | 
|  |  | 
|  | if (*first_arg) first_arg++;  /* skip program name */ | 
|  | cmdline = build_command_line( first_arg ); | 
|  |  | 
|  | if (WINE_TRACE_ON(winevdm)) | 
|  | { | 
|  | int i; | 
|  | WINE_TRACE( "GetCommandLine = '%s'\n", GetCommandLineA() ); | 
|  | WINE_TRACE( "appname = '%s'\n", appname ); | 
|  | WINE_TRACE( "cmdline = '%.*s'\n", cmdline[0], cmdline+1 ); | 
|  | for (i = 0; argv[i]; i++) WINE_TRACE( "argv[%d]: '%s'\n", i, argv[i] ); | 
|  | } | 
|  |  | 
|  | GetStartupInfoA( &info ); | 
|  | showCmd[0] = 2; | 
|  | showCmd[1] = (info.dwFlags & STARTF_USESHOWWINDOW) ? info.wShowWindow : SW_SHOWNORMAL; | 
|  |  | 
|  | params.hEnvironment = 0; | 
|  | params.cmdLine = MapLS( cmdline ); | 
|  | params.showCmd = MapLS( showCmd ); | 
|  | params.reserved = 0; | 
|  |  | 
|  | RestoreThunkLock(1);  /* grab the Win16 lock */ | 
|  |  | 
|  | /* some programs assume mmsystem is always present */ | 
|  | LoadLibrary16( "gdi.exe" ); | 
|  | LoadLibrary16( "user.exe" ); | 
|  | LoadLibrary16( "mmsystem.dll" ); | 
|  |  | 
|  | /* make sure system drivers are loaded */ | 
|  | LoadLibrary16( "comm.drv" ); | 
|  | LoadLibrary16( "display.drv" ); | 
|  | LoadLibrary16( "keyboard.drv" ); | 
|  | LoadLibrary16( "mouse.drv" ); | 
|  | LoadLibrary16( "sound.drv" ); | 
|  |  | 
|  | if ((instance = LoadModule16( appname, ¶ms )) < 32) | 
|  | { | 
|  | if (instance == 11) | 
|  | { | 
|  | /* first see if it is a .pif file */ | 
|  | if( ( p = strrchr( appname, '.' )) && !strcasecmp( p, ".pif")) | 
|  | pif_cmd( appname, cmdline + 1); | 
|  | else | 
|  | { | 
|  | if (!VirtualQuery( NULL, &mem_info, sizeof(mem_info) ) || mem_info.State == MEM_FREE) | 
|  | { | 
|  | WINE_MESSAGE( "winevdm: unable to exec '%s': DOS memory range unavailable\n", appname ); | 
|  | ExitProcess(1); | 
|  | } | 
|  |  | 
|  | /* try DOS format */ | 
|  | /* loader expects arguments to be regular C strings */ | 
|  | wine_load_dos_exe( appname, cmdline + 1 ); | 
|  | } | 
|  | /* if we get back here it failed */ | 
|  | instance = GetLastError(); | 
|  | } | 
|  |  | 
|  | WINE_MESSAGE( "winevdm: can't exec '%s': ", appname ); | 
|  | switch (instance) | 
|  | { | 
|  | case  2: WINE_MESSAGE("file not found\n" ); break; | 
|  | case 11: WINE_MESSAGE("invalid program file\n" ); break; | 
|  | default: WINE_MESSAGE("error=%d\n", instance ); break; | 
|  | } | 
|  | ExitProcess(instance); | 
|  | } | 
|  |  | 
|  | /* wait forever; the process will be killed when the last task exits */ | 
|  | ReleaseThunkLock( &count ); | 
|  | Sleep( INFINITE ); | 
|  | return 0; | 
|  | } |