1 /* 2 * COPYRIGHT: GPL - See COPYING in the top level directory 3 * PROJECT: ReactOS Virtual DOS Machine 4 * FILE: subsystems/mvdm/ntvdm/dos/dem.c 5 * PURPOSE: DOS 32-bit Emulation Support Library - 6 * This library is used by the built-in NTVDM DOS32 and by 7 * the NT 16-bit DOS in Windows (via BOPs). It also exposes 8 * exported functions that can be used by VDDs. 9 * PROGRAMMERS: Aleksandar Andrejevic <theflash AT sdf DOT lonestar DOT org> 10 * Hermes Belusca-Maito (hermes.belusca@sfr.fr) 11 */ 12 13 /* INCLUDES *******************************************************************/ 14 15 #include "ntvdm.h" 16 17 #define NDEBUG 18 #include <debug.h> 19 20 #include "emulator.h" 21 #include <isvbop.h> 22 23 #include "utils.h" 24 25 #include "dem.h" 26 #include "dos/dos32krnl/device.h" 27 #include "dos/dos32krnl/memory.h" 28 #include "dos/dos32krnl/process.h" 29 #include "cpu/bop.h" 30 #include "cpu/cpu.h" 31 32 #include "bios/bios.h" 33 #include "mouse32.h" 34 35 #include "vddsup.h" 36 37 /* 38 * EXPERIMENTAL! 39 * Activate this line if you want to have COMMAND.COM completely external. 40 */ 41 // #define COMSPEC_FULLY_EXTERNAL 42 43 /* PRIVATE VARIABLES **********************************************************/ 44 45 /* PRIVATE FUNCTIONS **********************************************************/ 46 47 /* PUBLIC VARIABLES ***********************************************************/ 48 49 /* PUBLIC FUNCTIONS ***********************************************************/ 50 51 52 /******************************************************************************\ 53 |** DOS DEM Kernel helpers **| 54 \******************************************************************************/ 55 56 57 VOID Dem_BiosCharPrint(CHAR Character) 58 { 59 /* Save AX and BX */ 60 USHORT AX = getAX(); 61 USHORT BX = getBX(); 62 63 /* 64 * Set the parameters: 65 * AL contains the character to print, 66 * BL contains the character attribute, 67 * BH contains the video page to use. 68 */ 69 setAL(Character); 70 setBL(DEFAULT_ATTRIBUTE); 71 setBH(Bda->VideoPage); 72 73 /* Call the BIOS INT 10h, AH=0Eh "Teletype Output" */ 74 setAH(0x0E); 75 Int32Call(&DosContext, BIOS_VIDEO_INTERRUPT); 76 77 /* Restore AX and BX */ 78 setBX(BX); 79 setAX(AX); 80 } 81 82 VOID DosCharPrint(CHAR Character) 83 { 84 DosPrintCharacter(DOS_OUTPUT_HANDLE, Character); 85 } 86 87 88 static VOID DemLoadNTDOSKernel(VOID) 89 { 90 BOOLEAN Success = FALSE; 91 LPCSTR DosKernelFileName = "ntdos.sys"; 92 HANDLE hDosKernel; 93 ULONG ulDosKernelSize = 0; 94 95 DPRINT1("You are loading Windows NT DOS!\n"); 96 97 /* Open the DOS kernel file */ 98 hDosKernel = FileOpen(DosKernelFileName, &ulDosKernelSize); 99 if (hDosKernel == NULL) goto Quit; 100 101 /* 102 * Attempt to load the DOS kernel into memory. 103 * The segment where to load the DOS kernel is defined 104 * by the DOS BIOS and is found in DI:0000 . 105 */ 106 Success = FileLoadByHandle(hDosKernel, 107 REAL_TO_PHYS(TO_LINEAR(getDI(), 0x0000)), 108 ulDosKernelSize, 109 &ulDosKernelSize); 110 111 DPRINT1("Windows NT DOS file '%s' loading %s at %04X:%04X, size 0x%X (Error: %u).\n", 112 DosKernelFileName, 113 (Success ? "succeeded" : "failed"), 114 getDI(), 0x0000, 115 ulDosKernelSize, 116 GetLastError()); 117 118 /* Close the DOS kernel file */ 119 FileClose(hDosKernel); 120 121 Quit: 122 if (!Success) 123 { 124 /* We failed everything, stop the VDM */ 125 BiosDisplayMessage("Windows NT DOS kernel file '%s' loading failed (Error: %u). The VDM will shut down.\n", 126 DosKernelFileName, GetLastError()); 127 EmulatorTerminate(); 128 return; 129 } 130 } 131 132 static VOID WINAPI DosSystemBop(LPWORD Stack) 133 { 134 /* Get the Function Number and skip it */ 135 BYTE FuncNum = *(PBYTE)SEG_OFF_TO_PTR(getCS(), getIP()); 136 setIP(getIP() + 1); 137 138 switch (FuncNum) 139 { 140 /* Load the DOS kernel */ 141 case 0x11: 142 { 143 DemLoadNTDOSKernel(); 144 break; 145 } 146 147 /* Call 32-bit Driver Strategy Routine */ 148 case BOP_DRV_STRATEGY: 149 { 150 DeviceStrategyBop(); 151 break; 152 } 153 154 /* Call 32-bit Driver Interrupt Routine */ 155 case BOP_DRV_INTERRUPT: 156 { 157 DeviceInterruptBop(); 158 break; 159 } 160 161 default: 162 { 163 DPRINT1("Unknown DOS System BOP Function: 0x%02X\n", FuncNum); 164 // setCF(1); // Disable, otherwise we enter an infinite loop 165 break; 166 } 167 } 168 } 169 170 171 172 173 /******************************************************************************\ 174 |** DOS Command Process management **| 175 \******************************************************************************/ 176 177 178 #ifndef STANDALONE 179 static ULONG SessionId = 0; 180 181 /* 182 * 16-bit Command Interpreter information for DOS reentry 183 */ 184 typedef struct _COMSPEC_INFO 185 { 186 LIST_ENTRY Entry; 187 DWORD dwExitCode; 188 WORD ComSpecPsp; 189 BOOLEAN Terminated; 190 } COMSPEC_INFO, *PCOMSPEC_INFO; 191 192 static COMSPEC_INFO RootCmd; 193 static DWORD ReentrancyCount = 0; 194 195 // FIXME: Should we need list locking? 196 static LIST_ENTRY ComSpecInfoList = { &ComSpecInfoList, &ComSpecInfoList }; 197 198 static PCOMSPEC_INFO 199 FindComSpecInfoByPsp(WORD Psp) 200 { 201 PLIST_ENTRY Pointer; 202 PCOMSPEC_INFO ComSpecInfo; 203 204 for (Pointer = ComSpecInfoList.Flink; Pointer != &ComSpecInfoList; Pointer = Pointer->Flink) 205 { 206 ComSpecInfo = CONTAINING_RECORD(Pointer, COMSPEC_INFO, Entry); 207 if (ComSpecInfo->ComSpecPsp == Psp) return ComSpecInfo; 208 } 209 210 return NULL; 211 } 212 213 static VOID 214 InsertComSpecInfo(PCOMSPEC_INFO ComSpecInfo) 215 { 216 InsertHeadList(&ComSpecInfoList, &ComSpecInfo->Entry); 217 } 218 219 static VOID 220 RemoveComSpecInfo(PCOMSPEC_INFO ComSpecInfo) 221 { 222 RemoveEntryList(&ComSpecInfo->Entry); 223 if (ComSpecInfo != &RootCmd) 224 RtlFreeHeap(RtlGetProcessHeap(), 0, ComSpecInfo); 225 } 226 #endif 227 228 static VOID DosProcessConsoleAttach(VOID) 229 { 230 /* Attach to the console */ 231 ConsoleAttach(); 232 VidBiosAttachToConsole(); 233 } 234 235 static VOID DosProcessConsoleDetach(VOID) 236 { 237 /* Detach from the console */ 238 VidBiosDetachFromConsole(); 239 ConsoleDetach(); 240 } 241 242 /* 243 * Data for the next DOS command to run 244 */ 245 #ifndef STANDALONE 246 static VDM_COMMAND_INFO CommandInfo; 247 static BOOLEAN Repeat = FALSE; 248 static BOOLEAN Reentry = FALSE; 249 #endif 250 static BOOLEAN First = TRUE; 251 static CHAR CmdLine[MAX_PATH] = ""; // DOS_CMDLINE_LENGTH 252 static CHAR AppName[MAX_PATH] = ""; 253 #ifndef STANDALONE 254 static CHAR PifFile[MAX_PATH] = ""; 255 static CHAR CurDirectory[MAX_PATH] = ""; 256 static CHAR Desktop[MAX_PATH] = ""; 257 static CHAR Title[MAX_PATH] = ""; 258 static ULONG EnvSize = 256; 259 static PVOID Env = NULL; 260 #endif 261 262 #pragma pack(push, 2) 263 264 /* 265 * This structure is compatible with Windows NT DOS 266 */ 267 typedef struct _NEXT_CMD 268 { 269 USHORT EnvBlockSeg; 270 USHORT EnvBlockLen; 271 USHORT CurDrive; 272 USHORT NumDrives; 273 USHORT CmdLineSeg; 274 USHORT CmdLineOff; 275 USHORT Unknown0; 276 USHORT ExitCode; 277 USHORT Unknown1; 278 ULONG Unknown2; 279 USHORT CodePage; 280 USHORT Unknown3; 281 USHORT Unknown4; 282 USHORT AppNameSeg; 283 USHORT AppNameOff; 284 USHORT AppNameLen; 285 USHORT Flags; 286 } NEXT_CMD, *PNEXT_CMD; 287 288 #pragma pack(pop) 289 290 static VOID CmdStartProcess(VOID) 291 { 292 #ifndef STANDALONE 293 PCOMSPEC_INFO ComSpecInfo; 294 #endif 295 SIZE_T CmdLen; 296 PNEXT_CMD DataStruct = (PNEXT_CMD)SEG_OFF_TO_PTR(getDS(), getDX()); 297 298 DPRINT1("CmdStartProcess -- DS:DX = %04X:%04X (DataStruct = 0x%p)\n", 299 getDS(), getDX(), DataStruct); 300 301 /* Pause the VM */ 302 EmulatorPause(); 303 304 #ifndef STANDALONE 305 /* Check whether we need to shell out now in case we were started by a 32-bit app */ 306 ComSpecInfo = FindComSpecInfoByPsp(Sda->CurrentPsp); 307 if (ComSpecInfo && ComSpecInfo->Terminated) 308 { 309 RemoveComSpecInfo(ComSpecInfo); 310 311 DPRINT1("Exit DOS from start-app BOP\n"); 312 setCF(1); 313 goto Quit; 314 } 315 316 /* Clear the structure */ 317 RtlZeroMemory(&CommandInfo, sizeof(CommandInfo)); 318 319 /* Initialize the structure members */ 320 CommandInfo.TaskId = SessionId; 321 CommandInfo.VDMState = VDM_FLAG_DOS; 322 CommandInfo.CmdLine = CmdLine; 323 CommandInfo.CmdLen = sizeof(CmdLine); 324 CommandInfo.AppName = AppName; 325 CommandInfo.AppLen = sizeof(AppName); 326 CommandInfo.PifFile = PifFile; 327 CommandInfo.PifLen = sizeof(PifFile); 328 CommandInfo.CurDirectory = CurDirectory; 329 CommandInfo.CurDirectoryLen = sizeof(CurDirectory); 330 CommandInfo.Desktop = Desktop; 331 CommandInfo.DesktopLen = sizeof(Desktop); 332 CommandInfo.Title = Title; 333 CommandInfo.TitleLen = sizeof(Title); 334 CommandInfo.Env = Env; 335 CommandInfo.EnvLen = EnvSize; 336 337 if (First) CommandInfo.VDMState |= VDM_FLAG_FIRST_TASK; 338 339 Command: 340 341 if (Repeat) CommandInfo.VDMState |= VDM_FLAG_RETRY; 342 Repeat = FALSE; 343 344 /* Get the VDM command information */ 345 DPRINT1("Calling GetNextVDMCommand in CmdStartProcess: wait for new VDM task...\n"); 346 if (!GetNextVDMCommand(&CommandInfo)) 347 { 348 DPRINT1("CmdStartProcess - GetNextVDMCommand failed, retrying... last error = %d\n", GetLastError()); 349 if (CommandInfo.EnvLen > EnvSize) 350 { 351 /* Expand the environment size */ 352 EnvSize = CommandInfo.EnvLen; 353 CommandInfo.Env = Env = RtlReAllocateHeap(RtlGetProcessHeap(), HEAP_ZERO_MEMORY, Env, EnvSize); 354 355 /* Repeat the request */ 356 Repeat = TRUE; 357 goto Command; 358 } 359 360 /* Shouldn't happen */ 361 DisplayMessage(L"An unrecoverable failure happened from start-app BOP; exiting DOS."); 362 setCF(1); 363 goto Quit; 364 } 365 366 // FIXME: What happens if some other 32-bit app is killed while we are waiting there?? 367 368 DPRINT1("CmdStartProcess - GetNextVDMCommand succeeded, start app...\n"); 369 370 #else 371 372 if (!First) 373 { 374 DPRINT1("Exit DOS from start-app BOP\n"); 375 setCF(1); 376 goto Quit; 377 } 378 379 #endif 380 381 /* Compute the command line length, not counting the terminating "\r\n" */ 382 CmdLen = strlen(CmdLine); 383 if (CmdLen >= 2 && CmdLine[CmdLen - 2] == '\r') 384 CmdLen -= 2; 385 386 DPRINT1("Starting '%s' ('%.*s')...\n", AppName, CmdLen, CmdLine); 387 388 /* Start the process */ 389 // FIXME: Merge 'Env' with the master environment SEG_OFF_TO_PTR(SYSTEM_ENV_BLOCK, 0) 390 // FIXME: Environment 391 RtlCopyMemory(SEG_OFF_TO_PTR(DataStruct->AppNameSeg, DataStruct->AppNameOff), AppName, MAX_PATH); 392 *(PBYTE)(SEG_OFF_TO_PTR(DataStruct->CmdLineSeg, DataStruct->CmdLineOff)) = (BYTE)CmdLen; 393 RtlCopyMemory(SEG_OFF_TO_PTR(DataStruct->CmdLineSeg, DataStruct->CmdLineOff + 1), CmdLine, DOS_CMDLINE_LENGTH); 394 395 #ifndef STANDALONE 396 /* Update console title if we run in a separate console */ 397 if (SessionId != 0) 398 SetConsoleTitleA(AppName); 399 #endif 400 401 First = FALSE; 402 setCF(0); 403 404 DPRINT1("App started!\n"); 405 406 Quit: 407 /* Resume the VM */ 408 EmulatorResume(); 409 } 410 411 static VOID CmdStartExternalCommand(VOID) 412 { 413 DWORD Result; 414 415 // TODO: improve: this code has strong similarities 416 // with the 'default' case of DosCreateProcess. 417 418 LPSTR Command = (LPSTR)SEG_OFF_TO_PTR(getDS(), getSI()); 419 CHAR CmdLine[sizeof("cmd.exe /c ") + DOS_CMDLINE_LENGTH + 1] = ""; 420 LPSTR CmdLinePtr; 421 SIZE_T CmdLineLen; 422 423 /* Spawn a user-defined 32-bit command preprocessor */ 424 425 // FIXME: Use COMSPEC env var!! 426 CmdLinePtr = CmdLine; 427 strcpy(CmdLinePtr, "cmd.exe /c "); 428 CmdLinePtr += strlen(CmdLinePtr); 429 430 /* Build a Win32-compatible command-line */ 431 CmdLineLen = min(strlen(Command), sizeof(CmdLine) - strlen(CmdLinePtr) - 1); 432 RtlCopyMemory(CmdLinePtr, Command, CmdLineLen); 433 CmdLinePtr[CmdLineLen] = '\0'; 434 435 /* Remove any trailing return carriage character and NULL-terminate the command line */ 436 while (*CmdLinePtr && *CmdLinePtr != '\r' && *CmdLinePtr != '\n') CmdLinePtr++; 437 *CmdLinePtr = '\0'; 438 439 DPRINT1("CMD Run Command '%s' ('%s')\n", Command, CmdLine); 440 441 /* 442 * No need to prepare the stack for DosStartComSpec since we won't start it. 443 */ 444 Result = DosStartProcess32(Command, CmdLine, 445 SEG_OFF_TO_PTR(getES(), 0) /*Environment*/, 446 MAKELONG(getIP(), getCS()) /*ReturnAddress*/, 447 FALSE); 448 if (Result != ERROR_SUCCESS) 449 { 450 DosDisplayMessage("Failed to start command '%s' ('%s'). Error: %u\n", Command, CmdLine, Result); 451 setCF(0); 452 setAL((UCHAR)Result); 453 } 454 else 455 { 456 DosDisplayMessage("Command '%s' ('%s') started successfully.\n", Command, CmdLine); 457 #ifndef STANDALONE 458 setCF(Repeat); // Set CF if we need to start a 16-bit process 459 #else 460 setCF(0); 461 #endif 462 } 463 } 464 465 static VOID CmdStartComSpec32(VOID) 466 { 467 DWORD Result; 468 469 // TODO: improve: this code has strong similarities with the 470 // 'default' case of DosCreateProcess and with the 'case 0x08'. 471 472 CHAR CmdLine[sizeof("cmd.exe") + 1] = ""; 473 474 /* Spawn a user-defined 32-bit command preprocessor */ 475 476 // FIXME: Use COMSPEC env var!! 477 strcpy(CmdLine, "cmd.exe"); 478 479 DPRINT1("CMD Run 32-bit Command Interpreter '%s'\n", CmdLine); 480 481 /* 482 * No need to prepare the stack for DosStartComSpec since we won't start it. 483 */ 484 Result = DosStartProcess32(CmdLine, CmdLine, 485 SEG_OFF_TO_PTR(getES(), 0) /*Environment*/, 486 MAKELONG(getIP(), getCS()) /*ReturnAddress*/, 487 FALSE); 488 if (Result != ERROR_SUCCESS) 489 { 490 DosDisplayMessage("Failed to start 32-bit Command Interpreter '%s'. Error: %u\n", CmdLine, Result); 491 setCF(0); 492 setAL((UCHAR)Result); 493 } 494 else 495 { 496 DosDisplayMessage("32-bit Command Interpreter '%s' started successfully.\n", CmdLine); 497 #ifndef STANDALONE 498 setCF(Repeat); // Set CF if we need to start a 16-bit process 499 #else 500 setCF(0); 501 #endif 502 } 503 } 504 505 static VOID CmdSetExitCode(VOID) 506 { 507 #ifndef STANDALONE 508 BOOL Success; 509 PCOMSPEC_INFO ComSpecInfo; 510 VDM_COMMAND_INFO CommandInfo; 511 #endif 512 513 /* Pause the VM */ 514 EmulatorPause(); 515 516 #ifndef STANDALONE 517 /* 518 * Check whether we need to shell out now in case we were started by a 32-bit app, 519 * or we were started alone along with the root 32-bit app. 520 */ 521 ComSpecInfo = FindComSpecInfoByPsp(Sda->CurrentPsp); 522 if ((ComSpecInfo && ComSpecInfo->Terminated) || 523 (ComSpecInfo == &RootCmd && SessionId != 0)) 524 { 525 RemoveComSpecInfo(ComSpecInfo); 526 #endif 527 DPRINT1("Exit DOS from ExitCode (prologue)!\n"); 528 setCF(0); 529 goto Quit; 530 #ifndef STANDALONE 531 } 532 533 /* Clear the VDM structure */ 534 RtlZeroMemory(&CommandInfo, sizeof(CommandInfo)); 535 536 Retry: 537 /* Update the VDM state of the task */ 538 // CommandInfo.TaskId = SessionId; 539 CommandInfo.ExitCode = getDX(); 540 CommandInfo.VDMState = VDM_FLAG_DONT_WAIT; 541 DPRINT1("Calling GetNextVDMCommand 32bit end of VDM task\n"); 542 Success = GetNextVDMCommand(&CommandInfo); 543 DPRINT1("GetNextVDMCommand 32bit end of VDM task success = %s, last error = %d\n", Success ? "true" : "false", GetLastError()); 544 545 /* 546 * Check whether we were awaited because the 32-bit process was stopped, 547 * or because it started a new DOS application. 548 */ 549 if (CommandInfo.CmdLen != 0 || CommandInfo.AppLen != 0 || CommandInfo.PifLen != 0) 550 { 551 DPRINT1("GetNextVDMCommand end-of-app, this is for a new VDM task - CmdLen = %d, AppLen = %d, PifLen = %d\n", 552 CommandInfo.CmdLen, CommandInfo.AppLen, CommandInfo.PifLen); 553 554 /* Repeat the request */ 555 Repeat = TRUE; 556 setCF(1); 557 } 558 else 559 { 560 DPRINT1("GetNextVDMCommand end-of-app, the app stopped\n"); 561 562 /* Check whether we need to shell out now in case we were started by a 32-bit app */ 563 ComSpecInfo = FindComSpecInfoByPsp(Sda->CurrentPsp); 564 if (!ComSpecInfo || !ComSpecInfo->Terminated) 565 { 566 DPRINT1("Not our 32-bit app, retrying...\n"); 567 goto Retry; 568 } 569 570 ASSERT(ComSpecInfo->Terminated == TRUE); 571 572 /* Record found, remove it and exit now */ 573 RemoveComSpecInfo(ComSpecInfo); 574 575 DPRINT1("Exit DOS from ExitCode wait!\n"); 576 setCF(0); 577 } 578 #endif 579 580 // FIXME: Use the retrieved exit code as the value of our exit code 581 // when COMMAND.COM will shell-out ?? 582 583 Quit: 584 /* Resume the VM */ 585 EmulatorResume(); 586 } 587 588 static VOID WINAPI DosCmdInterpreterBop(LPWORD Stack) 589 { 590 /* Get the Function Number and skip it */ 591 BYTE FuncNum = *(PBYTE)SEG_OFF_TO_PTR(getCS(), getIP()); 592 setIP(getIP() + 1); 593 594 switch (FuncNum) 595 { 596 /* Kill the VDM */ 597 case 0x00: 598 { 599 /* Stop the VDM */ 600 EmulatorTerminate(); 601 return; 602 } 603 604 /* 605 * Get a new app to start 606 * 607 * Input 608 * DS:DX : Data block. 609 * 610 * Output 611 * CF : 0: Success; 1: Failure. 612 */ 613 case 0x01: 614 { 615 CmdStartProcess(); 616 break; 617 } 618 619 /* 620 * Check binary format 621 * 622 * Input 623 * DS:DX : Program to check. 624 * 625 * Output 626 * CF : 0: Success; 1: Failure. 627 * AX : Error code. 628 */ 629 case 0x07: 630 { 631 DWORD BinaryType; 632 LPSTR ProgramName = (LPSTR)SEG_OFF_TO_PTR(getDS(), getDX()); 633 634 if (!GetBinaryTypeA(ProgramName, &BinaryType)) 635 { 636 /* An error happened, bail out */ 637 setCF(1); 638 setAX(LOWORD(GetLastError())); 639 break; 640 } 641 642 // FIXME: We only support DOS binaries for now... 643 ASSERT(BinaryType == SCS_DOS_BINARY); 644 if (BinaryType != SCS_DOS_BINARY) 645 { 646 /* An error happened, bail out */ 647 setCF(1); 648 setAX(LOWORD(ERROR_BAD_EXE_FORMAT)); 649 break; 650 } 651 652 /* Return success: DOS application */ 653 setCF(0); 654 break; 655 } 656 657 /* 658 * Start an external command 659 * 660 * Input 661 * DS:SI : Command to start. 662 * ES : Environment block segment. 663 * AL : Current drive number. 664 * AH : 0: Directly start the command; 665 * 1: Use "cmd.exe /c" to start the command. 666 * 667 * Output 668 * CF : 0: Shell-out; 1: Continue. 669 * AL : Error/Exit code. 670 */ 671 case 0x08: 672 { 673 CmdStartExternalCommand(); 674 break; 675 } 676 677 /* 678 * Start the default 32-bit command interpreter (COMSPEC) 679 * 680 * Input 681 * ES : Environment block segment. 682 * AL : Current drive number. 683 * 684 * Output 685 * CF : 0: Shell-out; 1: Continue. 686 * AL : Error/Exit code. 687 */ 688 case 0x0A: 689 { 690 CmdStartComSpec32(); 691 break; 692 } 693 694 /* 695 * Set exit code 696 * 697 * Input 698 * DX : Exit code 699 * 700 * Output 701 * CF : 0: Shell-out; 1: Continue. 702 */ 703 case 0x0B: 704 { 705 CmdSetExitCode(); 706 break; 707 } 708 709 /* 710 * Get start information 711 * 712 * Output 713 * AL : 0 (resp. 1): Started from (resp. without) an existing console. 714 */ 715 case 0x10: 716 { 717 #ifndef STANDALONE 718 /* 719 * When a new instance of our (internal) COMMAND.COM is started, 720 * we check whether we need to run a 32-bit COMSPEC. This goes by 721 * checking whether we were started in a new console (no parent 722 * console process) or from an existing one. 723 * 724 * However COMMAND.COM can also be started in the case where a 725 * 32-bit process (started by a 16-bit parent) wants to start a new 726 * 16-bit process: to ensure DOS reentry we need to start a new 727 * instance of COMMAND.COM. On Windows the COMMAND.COM is started 728 * just before the 32-bit process (in fact, it is this COMMAND.COM 729 * which starts the 32-bit process via an undocumented command-line 730 * switch '/z', which syntax is: 731 * COMMAND.COM /z\bAPPNAME.EXE 732 * notice the '\b' character inserted in-between. Then COMMAND.COM 733 * issues a BOP_CMD 08h with AH=00h to start the process). 734 * 735 * Instead, we do the reverse, i.e. we start the 32-bit process, 736 * and *only* if needed, i.e. if this process wants to start a 737 * new 16-bit process, we start our COMMAND.COM. 738 * 739 * The problem we then face is that our COMMAND.COM will possibly 740 * want to start a new COMSPEC, however we do not want this. 741 * The chosen solution is to flag this case -- done with the 'Reentry' 742 * boolean -- so that COMMAND.COM will not attempt to start COMSPEC 743 * but instead will directly try to start the 16-bit process. 744 */ 745 // setAL(SessionId != 0); 746 setAL((SessionId != 0) && !Reentry); 747 /* Reset 'Reentry' */ 748 Reentry = FALSE; 749 #else 750 setAL(0); 751 #endif 752 break; 753 } 754 755 default: 756 { 757 DPRINT1("Unknown DOS CMD Interpreter BOP Function: 0x%02X\n", FuncNum); 758 // setCF(1); // Disable, otherwise we enter an infinite loop 759 break; 760 } 761 } 762 } 763 764 #ifndef COMSPEC_FULLY_EXTERNAL 765 /* 766 * Internal COMMAND.COM binary data in the CommandCom array. 767 */ 768 #include "command_com.h" 769 #endif 770 771 static 772 DWORD DosStartComSpec(IN BOOLEAN Permanent, 773 IN LPCSTR Environment OPTIONAL, 774 IN DWORD ReturnAddress OPTIONAL, 775 OUT PWORD ComSpecPsp OPTIONAL) 776 { 777 /* 778 * BOOLEAN Permanent 779 * TRUE to simulate the /P switch of command.com: starts AUTOEXEC.BAT/NT 780 * and makes the interpreter permanent (cannot exit). 781 */ 782 783 DWORD Result; 784 785 if (ComSpecPsp) *ComSpecPsp = 0; 786 787 Result = 788 #ifndef COMSPEC_FULLY_EXTERNAL 789 DosLoadExecutableInternal(DOS_LOAD_AND_EXECUTE, 790 CommandCom, 791 sizeof(CommandCom), 792 "COMMAND.COM", 793 #else 794 DosLoadExecutable(DOS_LOAD_AND_EXECUTE, 795 #ifndef STANDALONE // FIXME: Those values are hardcoded paths on my local test machines!! 796 "C:\\CMDCMD.COM", 797 #else 798 "H:\\DOS_tests\\CMDCMD.COM", 799 #endif // STANDALONE 800 #endif // COMSPEC_FULLY_EXTERNAL 801 NULL, 802 Permanent ? "/P" : "", 803 Environment ? Environment : "", // FIXME: Default environment! 804 ReturnAddress); 805 if (Result != ERROR_SUCCESS) return Result; 806 807 /* TODO: Read AUTOEXEC.NT/BAT */ 808 809 /* Retrieve the PSP of the COMSPEC (current PSP set by DosLoadExecutable) */ 810 if (ComSpecPsp) *ComSpecPsp = Sda->CurrentPsp; 811 812 return Result; 813 } 814 815 typedef struct _DOS_START_PROC32 816 { 817 LPSTR ExecutablePath; 818 LPSTR CommandLine; 819 LPSTR Environment OPTIONAL; 820 #ifndef STANDALONE 821 PCOMSPEC_INFO ComSpecInfo; 822 HANDLE hEvent; 823 #endif 824 } DOS_START_PROC32, *PDOS_START_PROC32; 825 826 static DWORD 827 WINAPI 828 CommandThreadProc(LPVOID Parameter) 829 { 830 BOOL Success; 831 PROCESS_INFORMATION ProcessInfo; 832 STARTUPINFOA StartupInfo; 833 DWORD dwExitCode; 834 PDOS_START_PROC32 DosStartProc32 = (PDOS_START_PROC32)Parameter; 835 #ifndef STANDALONE 836 VDM_COMMAND_INFO CommandInfo; 837 PCOMSPEC_INFO ComSpecInfo = DosStartProc32->ComSpecInfo; 838 #endif 839 840 /* Set up the VDM, startup and process info structures */ 841 #ifndef STANDALONE 842 RtlZeroMemory(&CommandInfo, sizeof(CommandInfo)); 843 #endif 844 RtlZeroMemory(&ProcessInfo, sizeof(ProcessInfo)); 845 RtlZeroMemory(&StartupInfo, sizeof(StartupInfo)); 846 StartupInfo.cb = sizeof(StartupInfo); 847 848 // FIXME: Build suitable 32-bit environment!! 849 850 #ifndef STANDALONE 851 /* 852 * Wait for signaling a new VDM task and increment the VDM re-entry count so 853 * that we can handle 16-bit apps that may be possibly started by the 32-bit app. 854 */ 855 CommandInfo.VDMState = VDM_INC_REENTER_COUNT; 856 DPRINT1("Calling GetNextVDMCommand reenter++\n"); 857 Success = GetNextVDMCommand(&CommandInfo); 858 DPRINT1("GetNextVDMCommand reenter++ success = %s, last error = %d\n", Success ? "true" : "false", GetLastError()); 859 ++ReentrancyCount; 860 #endif 861 862 /* Start the process */ 863 Success = CreateProcessA(NULL, // ProgramName, 864 DosStartProc32->CommandLine, 865 NULL, 866 NULL, 867 TRUE, // Inherit handles 868 CREATE_DEFAULT_ERROR_MODE | CREATE_SUSPENDED, 869 DosStartProc32->Environment, 870 NULL, // lpCurrentDirectory, see "START" command in cmd.exe 871 &StartupInfo, 872 &ProcessInfo); 873 874 #ifndef STANDALONE 875 /* Signal our caller the process was started */ 876 SetEvent(DosStartProc32->hEvent); 877 // After this point, 'DosStartProc32' is not valid anymore. 878 #endif 879 880 if (Success) 881 { 882 /* Resume the process */ 883 ResumeThread(ProcessInfo.hThread); 884 885 /* Wait for the process to finish running and retrieve its exit code */ 886 WaitForSingleObject(ProcessInfo.hProcess, INFINITE); 887 GetExitCodeProcess(ProcessInfo.hProcess, &dwExitCode); 888 889 /* Close the handles */ 890 CloseHandle(ProcessInfo.hThread); 891 CloseHandle(ProcessInfo.hProcess); 892 } 893 else 894 { 895 dwExitCode = GetLastError(); 896 } 897 898 #ifndef STANDALONE 899 ASSERT(ComSpecInfo); 900 ComSpecInfo->Terminated = TRUE; 901 ComSpecInfo->dwExitCode = dwExitCode; 902 903 /* Decrement the VDM re-entry count */ 904 CommandInfo.VDMState = VDM_DEC_REENTER_COUNT; 905 DPRINT1("Calling GetNextVDMCommand reenter--\n"); 906 Success = GetNextVDMCommand(&CommandInfo); 907 DPRINT1("GetNextVDMCommand reenter-- success = %s, last error = %d\n", Success ? "true" : "false", GetLastError()); 908 --ReentrancyCount; 909 910 return 0; 911 #else 912 return dwExitCode; 913 #endif 914 } 915 916 DWORD DosStartProcess32(IN LPCSTR ExecutablePath, 917 IN LPCSTR CommandLine, 918 IN LPCSTR Environment OPTIONAL, 919 IN DWORD ReturnAddress OPTIONAL, 920 IN BOOLEAN StartComSpec) 921 { 922 DWORD Result = ERROR_SUCCESS; 923 HANDLE CommandThread; 924 DOS_START_PROC32 DosStartProc32; 925 #ifndef STANDALONE 926 BOOL Success; 927 VDM_COMMAND_INFO CommandInfo; 928 #endif 929 930 DosStartProc32.ExecutablePath = (LPSTR)ExecutablePath; 931 DosStartProc32.CommandLine = (LPSTR)CommandLine; 932 DosStartProc32.Environment = (LPSTR)Environment; 933 934 #ifndef STANDALONE 935 DosStartProc32.ComSpecInfo = 936 RtlAllocateHeap(RtlGetProcessHeap(), 937 HEAP_ZERO_MEMORY, 938 sizeof(*DosStartProc32.ComSpecInfo)); 939 ASSERT(DosStartProc32.ComSpecInfo); 940 941 DosStartProc32.hEvent = CreateEventW(NULL, FALSE, FALSE, NULL); 942 ASSERT(DosStartProc32.hEvent); 943 #endif 944 945 /* Pause the VM and detach from the console */ 946 EmulatorPause(); 947 DosProcessConsoleDetach(); 948 949 /* Start the 32-bit process via another thread */ 950 CommandThread = CreateThread(NULL, 0, &CommandThreadProc, &DosStartProc32, 0, NULL); 951 if (CommandThread == NULL) 952 { 953 DisplayMessage(L"FATAL: Failed to create the command processing thread: %d", GetLastError()); 954 Result = GetLastError(); 955 goto Quit; 956 } 957 958 #ifndef STANDALONE 959 /* Close the thread handle */ 960 CloseHandle(CommandThread); 961 962 /* Wait for the process to be ready to start */ 963 WaitForSingleObject(DosStartProc32.hEvent, INFINITE); 964 965 /* Wait for any potential new DOS app started by the 32-bit process */ 966 RtlZeroMemory(&CommandInfo, sizeof(CommandInfo)); 967 968 Retry: 969 CommandInfo.VDMState = VDM_FLAG_NESTED_TASK | VDM_FLAG_DONT_WAIT; 970 DPRINT1("Calling GetNextVDMCommand 32bit for possible new VDM task...\n"); 971 Success = GetNextVDMCommand(&CommandInfo); 972 DPRINT1("GetNextVDMCommand 32bit awaited, success = %s, last error = %d\n", Success ? "true" : "false", GetLastError()); 973 974 /* 975 * Check whether we were awaited because the 32-bit process was stopped, 976 * or because it started a new DOS application. 977 */ 978 if (CommandInfo.CmdLen != 0 || CommandInfo.AppLen != 0 || CommandInfo.PifLen != 0) 979 { 980 DPRINT1("GetNextVDMCommand 32bit, this is for a new VDM task - CmdLen = %d, AppLen = %d, PifLen = %d\n", 981 CommandInfo.CmdLen, CommandInfo.AppLen, CommandInfo.PifLen); 982 983 /* Repeat the request */ 984 Repeat = TRUE; 985 986 /* 987 * Set 'Reentry' to TRUE or FALSE depending on whether we are going 988 * to reenter with a new COMMAND.COM. See the comment for: 989 * BOP_CMD 0x10 'Get start information' 990 * (dem.c!DosCmdInterpreterBop) for more details. 991 */ 992 Reentry = StartComSpec; 993 994 /* If needed, start a new command interpreter to handle the possible incoming DOS commands */ 995 if (StartComSpec) 996 { 997 // 998 // DosStartProcess32 was only called by DosCreateProcess, called from INT 21h, 999 // so the caller stack is already prepared for running a new DOS program 1000 // (Flags, CS and IP, and the extra interrupt number, are already pushed). 1001 // 1002 Result = DosStartComSpec(FALSE, Environment, ReturnAddress, 1003 &DosStartProc32.ComSpecInfo->ComSpecPsp); 1004 if (Result != ERROR_SUCCESS) 1005 { 1006 DosDisplayMessage("Failed to start a new Command Interpreter (Error: %u).\n", Result); 1007 goto Quit; 1008 } 1009 } 1010 else 1011 { 1012 /* Retrieve the PSP of the COMSPEC (current PSP set by DosLoadExecutable) */ 1013 DosStartProc32.ComSpecInfo->ComSpecPsp = Sda->CurrentPsp; 1014 Result = ERROR_SUCCESS; 1015 } 1016 1017 /* Insert the new entry in the list; it will be freed when needed by COMMAND.COM */ 1018 InsertComSpecInfo(DosStartProc32.ComSpecInfo); 1019 } 1020 else 1021 { 1022 DPRINT1("GetNextVDMCommand 32bit, 32-bit app stopped\n"); 1023 1024 /* Check whether this was our 32-bit app which was killed */ 1025 if (!DosStartProc32.ComSpecInfo->Terminated) 1026 { 1027 DPRINT1("Not our 32-bit app, retrying...\n"); 1028 goto Retry; 1029 } 1030 1031 Result = DosStartProc32.ComSpecInfo->dwExitCode; 1032 1033 /* Delete the entry */ 1034 RtlFreeHeap(RtlGetProcessHeap(), 0, DosStartProc32.ComSpecInfo); 1035 } 1036 #else 1037 /* Wait for the thread to finish */ 1038 WaitForSingleObject(CommandThread, INFINITE); 1039 GetExitCodeThread(CommandThread, &Result); 1040 1041 /* Close the thread handle */ 1042 CloseHandle(CommandThread); 1043 1044 DPRINT1("32-bit app stopped\n"); 1045 #endif 1046 1047 Quit: 1048 #ifndef STANDALONE 1049 CloseHandle(DosStartProc32.hEvent); 1050 #endif 1051 1052 /* Attach to the console and resume the VM */ 1053 DosProcessConsoleAttach(); 1054 EmulatorResume(); 1055 1056 return Result; 1057 } 1058 1059 1060 1061 1062 /******************************************************************************\ 1063 |** DOS Bootloader emulation, Startup and Shutdown **| 1064 \******************************************************************************/ 1065 1066 1067 // 1068 // This function (equivalent of the DOS bootsector) is called by the bootstrap 1069 // loader *BEFORE* jumping at 0000:7C00. What we should do is to write at 0000:7C00 1070 // a BOP call that calls DosInitialize back. Then the bootstrap loader jumps at 1071 // 0000:7C00, our BOP gets called and then we can initialize the 32-bit part of the DOS. 1072 // 1073 1074 /* 16-bit bootstrap code at 0000:7C00 */ 1075 /* Of course, this is not in real bootsector format, because we don't care about it for now */ 1076 static BYTE Bootsector1[] = 1077 { 1078 LOBYTE(EMULATOR_BOP), HIBYTE(EMULATOR_BOP), BOP_LOAD_DOS 1079 }; 1080 /* This portion of code is run if we failed to load the DOS */ 1081 // NOTE: This may also be done by the BIOS32. 1082 static BYTE Bootsector2[] = 1083 { 1084 LOBYTE(EMULATOR_BOP), HIBYTE(EMULATOR_BOP), BOP_UNSIMULATE 1085 }; 1086 1087 static VOID WINAPI DosInitialize(LPWORD Stack); 1088 1089 VOID DosBootsectorInitialize(VOID) 1090 { 1091 /* We write the bootsector at 0000:7C00 */ 1092 ULONG_PTR StartAddress = (ULONG_PTR)SEG_OFF_TO_PTR(0x0000, 0x7C00); 1093 ULONG_PTR Address = StartAddress; 1094 CHAR DosKernelFileName[] = ""; // No DOS BIOS file name, therefore we will load DOS32 1095 1096 DPRINT("DosBootsectorInitialize\n"); 1097 1098 /* Write the "bootsector" */ 1099 RtlCopyMemory((PVOID)Address, Bootsector1, sizeof(Bootsector1)); 1100 Address += sizeof(Bootsector1); 1101 RtlCopyMemory((PVOID)Address, DosKernelFileName, sizeof(DosKernelFileName)); 1102 Address += sizeof(DosKernelFileName); 1103 RtlCopyMemory((PVOID)Address, Bootsector2, sizeof(Bootsector2)); 1104 Address += sizeof(Bootsector2); 1105 1106 /* Initialize the callback context */ 1107 InitializeContext(&DosContext, 0x0000, 1108 (ULONG_PTR)MEM_ALIGN_UP(0x7C00 + Address - StartAddress, sizeof(WORD))); 1109 1110 /* Register the DOS Loading BOP */ 1111 RegisterBop(BOP_LOAD_DOS, DosInitialize); 1112 } 1113 1114 1115 // 1116 // This function is called by the DOS bootsector in case we load DOS32. 1117 // It sets up the DOS32 start code then jumps to 0070:0000. 1118 // 1119 1120 /* 16-bit startup code for DOS32 at 0070:0000 */ 1121 static BYTE Startup[] = 1122 { 1123 LOBYTE(EMULATOR_BOP), HIBYTE(EMULATOR_BOP), BOP_START_DOS, 1124 LOBYTE(EMULATOR_BOP), HIBYTE(EMULATOR_BOP), BOP_UNSIMULATE 1125 }; 1126 1127 static VOID WINAPI DosStart(LPWORD Stack); 1128 1129 static VOID WINAPI DosInitialize(LPWORD Stack) 1130 { 1131 /* Get the DOS BIOS file name (NULL-terminated) */ 1132 // FIXME: Isn't it possible to use some DS:SI instead?? 1133 LPCSTR DosBiosFileName = (LPCSTR)SEG_OFF_TO_PTR(getCS(), getIP()); 1134 setIP(getIP() + (USHORT)strlen(DosBiosFileName) + 1); // Skip it 1135 1136 DPRINT("DosInitialize('%s')\n", DosBiosFileName); 1137 1138 /* 1139 * We succeeded, deregister the DOS Loading BOP 1140 * so that no app will be able to call us back. 1141 */ 1142 RegisterBop(BOP_LOAD_DOS, NULL); 1143 1144 /* Register the DOS BOPs */ 1145 RegisterBop(BOP_DOS, DosSystemBop ); 1146 RegisterBop(BOP_CMD, DosCmdInterpreterBop); 1147 1148 if (DosBiosFileName[0] != '\0') 1149 { 1150 BOOLEAN Success = FALSE; 1151 HANDLE hDosBios; 1152 ULONG ulDosBiosSize = 0; 1153 1154 /* Open the DOS BIOS file */ 1155 hDosBios = FileOpen(DosBiosFileName, &ulDosBiosSize); 1156 if (hDosBios == NULL) goto Quit; 1157 1158 /* Attempt to load the DOS BIOS into memory */ 1159 Success = FileLoadByHandle(hDosBios, 1160 REAL_TO_PHYS(TO_LINEAR(0x0070, 0x0000)), 1161 ulDosBiosSize, 1162 &ulDosBiosSize); 1163 1164 DPRINT1("DOS BIOS file '%s' loading %s at %04X:%04X, size 0x%X (Error: %u).\n", 1165 DosBiosFileName, 1166 (Success ? "succeeded" : "failed"), 1167 0x0070, 0x0000, 1168 ulDosBiosSize, 1169 GetLastError()); 1170 1171 /* Close the DOS BIOS file */ 1172 FileClose(hDosBios); 1173 1174 Quit: 1175 if (!Success) 1176 { 1177 BiosDisplayMessage("DOS BIOS file '%s' loading failed (Error: %u). The VDM will shut down.\n", 1178 DosBiosFileName, GetLastError()); 1179 return; 1180 } 1181 } 1182 else 1183 { 1184 /* Load the 16-bit startup code for DOS32 and register its Starting BOP */ 1185 RtlCopyMemory(SEG_OFF_TO_PTR(0x0070, 0x0000), Startup, sizeof(Startup)); 1186 1187 // This is the equivalent of BOP_LOAD_DOS, function 0x11 "Load the DOS kernel" 1188 // for the Windows NT DOS. 1189 RegisterBop(BOP_START_DOS, DosStart); 1190 } 1191 1192 /* Position execution pointers for DOS startup and return */ 1193 setCS(0x0070); 1194 setIP(0x0000); 1195 } 1196 1197 static VOID WINAPI DosStart(LPWORD Stack) 1198 { 1199 BOOLEAN Success; 1200 DWORD Result; 1201 #ifndef STANDALONE 1202 INT i; 1203 #endif 1204 1205 DPRINT("DosStart\n"); 1206 1207 /* 1208 * We succeeded, deregister the DOS Starting BOP 1209 * so that no app will be able to call us back. 1210 */ 1211 RegisterBop(BOP_START_DOS, NULL); 1212 1213 /* Initialize the callback context */ 1214 InitializeContext(&DosContext, BIOS_CODE_SEGMENT, 0x0010); 1215 1216 Success = DosBIOSInitialize(); 1217 // Success &= DosKRNLInitialize(); 1218 if (!Success) 1219 { 1220 BiosDisplayMessage("DOS32 loading failed (Error: %u). The VDM will shut down.\n", GetLastError()); 1221 EmulatorTerminate(); 1222 return; 1223 } 1224 1225 /* Load the mouse driver */ 1226 DosMouseInitialize(); 1227 1228 #ifndef STANDALONE 1229 1230 /* Parse the command line arguments */ 1231 for (i = 1; i < NtVdmArgc; i++) 1232 { 1233 if (wcsncmp(NtVdmArgv[i], L"-i", 2) == 0) 1234 { 1235 /* This is the session ID (hex format) */ 1236 SessionId = wcstoul(NtVdmArgv[i] + 2, NULL, 16); 1237 } 1238 } 1239 1240 /* Initialize Win32-VDM environment */ 1241 Env = RtlAllocateHeap(RtlGetProcessHeap(), HEAP_ZERO_MEMORY, EnvSize); 1242 if (Env == NULL) 1243 { 1244 DosDisplayMessage("Failed to initialize the global environment (Error: %u). The VDM will shut down.\n", GetLastError()); 1245 EmulatorTerminate(); 1246 return; 1247 } 1248 1249 /* Clear the structure */ 1250 RtlZeroMemory(&CommandInfo, sizeof(CommandInfo)); 1251 1252 /* Get the initial information */ 1253 CommandInfo.TaskId = SessionId; 1254 CommandInfo.VDMState = VDM_GET_FIRST_COMMAND | VDM_FLAG_DOS; 1255 GetNextVDMCommand(&CommandInfo); 1256 1257 #else 1258 1259 /* Retrieve the command to start */ 1260 if (NtVdmArgc >= 2) 1261 { 1262 WideCharToMultiByte(CP_ACP, 0, NtVdmArgv[1], -1, AppName, sizeof(AppName), NULL, NULL); 1263 1264 if (NtVdmArgc >= 3) 1265 WideCharToMultiByte(CP_ACP, 0, NtVdmArgv[2], -1, CmdLine, sizeof(CmdLine), NULL, NULL); 1266 else 1267 strcpy(CmdLine, ""); 1268 } 1269 else 1270 { 1271 DosDisplayMessage("Invalid DOS command line\n"); 1272 EmulatorTerminate(); 1273 return; 1274 } 1275 1276 #endif 1277 1278 /* 1279 * At this point, CS:IP points to the DOS BIOS exit code. If the 1280 * root command interpreter fails to start (or if it exits), DOS 1281 * exits and the VDM terminates. 1282 */ 1283 1284 /* Start the root command interpreter */ 1285 // TODO: Really interpret the 'SHELL=' line of CONFIG.NT, and use it! 1286 1287 /* 1288 * Prepare the stack for DosStartComSpec: 1289 * push Flags, CS and IP, and an extra WORD. 1290 */ 1291 setSP(getSP() - sizeof(WORD)); 1292 *((LPWORD)SEG_OFF_TO_PTR(getSS(), getSP())) = (WORD)getEFLAGS(); 1293 setSP(getSP() - sizeof(WORD)); 1294 *((LPWORD)SEG_OFF_TO_PTR(getSS(), getSP())) = getCS(); 1295 setSP(getSP() - sizeof(WORD)); 1296 *((LPWORD)SEG_OFF_TO_PTR(getSS(), getSP())) = getIP(); 1297 setSP(getSP() - sizeof(WORD)); 1298 1299 Result = DosStartComSpec(TRUE, SEG_OFF_TO_PTR(SYSTEM_ENV_BLOCK, 0), 1300 MAKELONG(getIP(), getCS()), 1301 #ifndef STANDALONE 1302 &RootCmd.ComSpecPsp 1303 #else 1304 NULL 1305 #endif 1306 ); 1307 if (Result != ERROR_SUCCESS) 1308 { 1309 /* Unprepare the stack for DosStartComSpec */ 1310 setSP(getSP() + 4*sizeof(WORD)); 1311 1312 DosDisplayMessage("Failed to start the Command Interpreter (Error: %u). The VDM will shut down.\n", Result); 1313 EmulatorTerminate(); 1314 return; 1315 } 1316 1317 #ifndef STANDALONE 1318 RootCmd.Terminated = FALSE; 1319 InsertComSpecInfo(&RootCmd); 1320 #endif 1321 1322 /**/ 1323 /* Attach to the console and resume the VM */ 1324 DosProcessConsoleAttach(); 1325 EmulatorResume(); 1326 /**/ 1327 1328 return; 1329 } 1330 1331 BOOLEAN DosShutdown(BOOLEAN Immediate) 1332 { 1333 /* 1334 * Immediate = TRUE: Immediate shutdown; 1335 * FALSE: Delayed shutdown (notification). 1336 */ 1337 1338 #ifndef STANDALONE 1339 if (Immediate) 1340 { 1341 ExitVDM(FALSE, 0); 1342 return TRUE; 1343 } 1344 else 1345 { 1346 // HACK! 1347 extern HANDLE VdmTaskEvent; // see emulator.c 1348 1349 /* 1350 * Signal the root COMMAND.COM that it should terminate 1351 * when it checks for a new command. 1352 */ 1353 RootCmd.Terminated = TRUE; 1354 1355 /* If the list is already empty, or just contains only one element, bail out */ 1356 // FIXME: Question: if the list has only one element, is it ALWAYS RootCmd ?? 1357 // FIXME: The following is hackish. 1358 if ((IsListEmpty(&ComSpecInfoList) || 1359 (ComSpecInfoList.Flink == &RootCmd.Entry && 1360 ComSpecInfoList.Blink == &RootCmd.Entry)) && 1361 ReentrancyCount == 0 && 1362 WaitForSingleObject(VdmTaskEvent, 0) == WAIT_TIMEOUT) 1363 { 1364 /* Nothing runs, so exit immediately */ 1365 ExitVDM(FALSE, 0); 1366 return TRUE; 1367 } 1368 1369 return FALSE; 1370 } 1371 #else 1372 UNREFERENCED_PARAMETER(Immediate); 1373 return TRUE; 1374 #endif 1375 } 1376 1377 1378 /* PUBLIC EXPORTED APIS *******************************************************/ 1379 1380 // demLFNCleanup 1381 // demLFNGetCurrentDirectory 1382 1383 // demGetFileTimeByHandle_WOW 1384 // demWOWLFNAllocateSearchHandle 1385 // demWOWLFNCloseSearchHandle 1386 // demWOWLFNEntry 1387 // demWOWLFNGetSearchHandle 1388 // demWOWLFNInit 1389 1390 DWORD 1391 WINAPI 1392 demClientErrorEx(IN HANDLE FileHandle, 1393 IN CHAR Unknown, 1394 IN BOOL Flag) 1395 { 1396 UNIMPLEMENTED; 1397 return GetLastError(); 1398 } 1399 1400 DWORD 1401 WINAPI 1402 demFileDelete(IN LPCSTR FileName) 1403 { 1404 if (DeleteFileA(FileName)) SetLastError(ERROR_SUCCESS); 1405 1406 return GetLastError(); 1407 } 1408 1409 /** 1410 * @brief Helper for demFileFindFirst() and demFileFindNext(). 1411 * Returns TRUE if a file matches the DOS attributes and has a 8.3 file name. 1412 **/ 1413 static BOOLEAN 1414 dempIsFileMatch( 1415 _Inout_ PWIN32_FIND_DATAA FindData, 1416 _In_ WORD AttribMask, 1417 _Out_ PCSTR* ShortName) 1418 { 1419 /* Convert in place the attributes to DOS format */ 1420 FindData->dwFileAttributes = NT_TO_DOS_FA(FindData->dwFileAttributes); 1421 1422 /* Check the attributes */ 1423 if ((FindData->dwFileAttributes & (FA_HIDDEN | FA_SYSTEM | FA_DIRECTORY)) 1424 & ~AttribMask) 1425 { 1426 return FALSE; 1427 } 1428 1429 /* Check whether the file has a 8.3 file name */ 1430 if (*FindData->cAlternateFileName) 1431 { 1432 /* Use the available one */ 1433 *ShortName = FindData->cAlternateFileName; 1434 return TRUE; 1435 } 1436 else 1437 { 1438 /* 1439 * Verify whether the original long name is actually a valid 1440 * 8.3 file name. Note that we cannot use GetShortPathName() 1441 * since the latter works on full paths, that we do not always have. 1442 */ 1443 BOOLEAN IsNameLegal, SpacesInName; 1444 WCHAR FileNameBufferU[_countof(FindData->cFileName) + 1]; 1445 UNICODE_STRING FileNameU; 1446 ANSI_STRING FileNameA; 1447 1448 RtlInitAnsiString(&FileNameA, FindData->cFileName); 1449 RtlInitEmptyUnicodeString(&FileNameU, FileNameBufferU, sizeof(FileNameBufferU)); 1450 RtlAnsiStringToUnicodeString(&FileNameU, &FileNameA, FALSE); 1451 1452 IsNameLegal = RtlIsNameLegalDOS8Dot3(&FileNameU, 1453 NULL, // (lpOemName ? &AnsiName : NULL), 1454 &SpacesInName); 1455 1456 if (!IsNameLegal || SpacesInName) 1457 { 1458 /* This is an error situation */ 1459 DPRINT1("'%.*s' is %s 8.3 filename %s spaces\n", 1460 _countof(FindData->cFileName), FindData->cFileName, 1461 (IsNameLegal ? "a valid" : "an invalid"), (SpacesInName ? "with" : "without")); 1462 } 1463 1464 if (IsNameLegal && !SpacesInName) 1465 { 1466 /* We can use the original name */ 1467 *ShortName = FindData->cFileName; 1468 return TRUE; 1469 } 1470 } 1471 1472 DPRINT1("No short 8.3 filename available for '%.*s'\n", 1473 _countof(FindData->cFileName), FindData->cFileName); 1474 1475 return FALSE; 1476 } 1477 1478 /** 1479 * @name demFileFindFirst 1480 * Implementation of the DOS INT 21h, AH=4Eh "Find First File" function. 1481 * 1482 * Starts enumerating files that match the given file search specification 1483 * and whose attributes are _at most_ those specified by the mask. This means 1484 * in particular that "normal files", i.e. files with no attributes set, are 1485 * always enumerated along those matching the requested attributes. 1486 * 1487 * @param[out] pFindFileData 1488 * Pointer to the DTA (Disk Transfer Area) filled with FindFirst data block. 1489 * 1490 * @param[in] FileName 1491 * File search specification (may include path and wildcards). 1492 * 1493 * @param[in] AttribMask 1494 * Mask of file attributes. Includes files with a given attribute bit set 1495 * if the corresponding bit is set to 1 in the mask. Excludes files with a 1496 * given attribute bit set if the corresponding bit is set to 0 in the mask. 1497 * Supported file attributes: 1498 * FA_NORMAL 0x0000 1499 * FA_READONLY 0x0001 (ignored) 1500 * FA_HIDDEN 0x0002 1501 * FA_SYSTEM 0x0004 1502 * FA_VOLID 0x0008 (not currently supported) 1503 * FA_LABEL 1504 * FA_DIRECTORY 0x0010 1505 * FA_ARCHIVE 0x0020 (ignored) 1506 * FA_DEVICE 0x0040 (ignored) 1507 * 1508 * @return 1509 * ERROR_SUCCESS on success (found match), or a last error (match not found). 1510 * 1511 * @see demFileFindNext() 1512 **/ 1513 DWORD 1514 WINAPI 1515 demFileFindFirst( 1516 _Out_ PVOID pFindFileData, 1517 _In_ PCSTR FileName, 1518 _In_ WORD AttribMask) 1519 { 1520 PDOS_FIND_FILE_BLOCK FindFileBlock = (PDOS_FIND_FILE_BLOCK)pFindFileData; 1521 HANDLE SearchHandle; 1522 WIN32_FIND_DATAA FindData; 1523 PCSTR ShortName = NULL; 1524 1525 /* Reset the private block fields */ 1526 RtlZeroMemory(FindFileBlock, RTL_SIZEOF_THROUGH_FIELD(DOS_FIND_FILE_BLOCK, SearchHandle)); 1527 1528 // TODO: Handle FA_VOLID for volume label. 1529 if (AttribMask & FA_VOLID) 1530 { 1531 DPRINT1("demFileFindFirst: Volume label attribute is UNIMPLEMENTED!\n"); 1532 AttribMask &= ~FA_VOLID; // Remove it for the time being... 1533 } 1534 1535 /* Filter out the ignored attributes */ 1536 AttribMask &= ~(FA_DEVICE | FA_ARCHIVE | FA_READONLY); 1537 1538 /* Start a search */ 1539 SearchHandle = FindFirstFileA(FileName, &FindData); 1540 if (SearchHandle == INVALID_HANDLE_VALUE) 1541 return GetLastError(); 1542 1543 /* Check the attributes and retry as long as we haven't found a matching file */ 1544 while (!dempIsFileMatch(&FindData, AttribMask, &ShortName)) 1545 { 1546 /* Continue searching. If we fail at some point, 1547 * stop the search and return an error. */ 1548 if (!FindNextFileA(SearchHandle, &FindData)) 1549 { 1550 FindClose(SearchHandle); 1551 return GetLastError(); 1552 } 1553 } 1554 1555 /* Fill the block */ 1556 FindFileBlock->DriveLetter = DosData->Sda.CurrentDrive + 'A'; 1557 strncpy(FindFileBlock->Pattern, FileName, _countof(FindFileBlock->Pattern)); 1558 FindFileBlock->AttribMask = AttribMask; 1559 FindFileBlock->SearchHandle = SearchHandle; 1560 FindFileBlock->Attributes = LOBYTE(FindData.dwFileAttributes); 1561 FileTimeToDosDateTime(&FindData.ftLastWriteTime, 1562 &FindFileBlock->FileDate, 1563 &FindFileBlock->FileTime); 1564 FindFileBlock->FileSize = FindData.nFileSizeHigh ? MAXDWORD 1565 : FindData.nFileSizeLow; 1566 1567 /* Copy the NULL-terminated short file name */ 1568 RtlStringCchCopyA(FindFileBlock->FileName, 1569 _countof(FindFileBlock->FileName), 1570 ShortName); 1571 1572 return ERROR_SUCCESS; 1573 } 1574 1575 /** 1576 * @name demFileFindNext 1577 * Implementation of the DOS INT 21h, AH=4Fh "Find Next File" function. 1578 * 1579 * Continues enumerating files, with the same file search specification 1580 * and attributes as those given to the first demFileFindFirst() call. 1581 * 1582 * @param[in,out] pFindFileData 1583 * Pointer to the DTA (Disk Transfer Area) filled with FindFirst data block. 1584 * 1585 * @return 1586 * ERROR_SUCCESS on success (found match), or a last error (match not found). 1587 * 1588 * @see demFileFindFirst() 1589 **/ 1590 DWORD 1591 WINAPI 1592 demFileFindNext( 1593 _Inout_ PVOID pFindFileData) 1594 { 1595 PDOS_FIND_FILE_BLOCK FindFileBlock = (PDOS_FIND_FILE_BLOCK)pFindFileData; 1596 HANDLE SearchHandle = FindFileBlock->SearchHandle; 1597 WIN32_FIND_DATAA FindData; 1598 PCSTR ShortName = NULL; 1599 1600 do 1601 { 1602 /* Continue searching. If we fail at some point, 1603 * stop the search and return an error. */ 1604 if (!FindNextFileA(SearchHandle, &FindData)) 1605 { 1606 FindClose(SearchHandle); 1607 1608 /* Reset the private block fields */ 1609 RtlZeroMemory(FindFileBlock, RTL_SIZEOF_THROUGH_FIELD(DOS_FIND_FILE_BLOCK, SearchHandle)); 1610 return GetLastError(); 1611 } 1612 } 1613 /* Check the attributes and retry as long as we haven't found a matching file */ 1614 while (!dempIsFileMatch(&FindData, FindFileBlock->AttribMask, &ShortName)); 1615 1616 /* Update the block */ 1617 FindFileBlock->Attributes = LOBYTE(FindData.dwFileAttributes); 1618 FileTimeToDosDateTime(&FindData.ftLastWriteTime, 1619 &FindFileBlock->FileDate, 1620 &FindFileBlock->FileTime); 1621 FindFileBlock->FileSize = FindData.nFileSizeHigh ? MAXDWORD 1622 : FindData.nFileSizeLow; 1623 1624 /* Copy the NULL-terminated short file name */ 1625 RtlStringCchCopyA(FindFileBlock->FileName, 1626 _countof(FindFileBlock->FileName), 1627 ShortName); 1628 1629 return ERROR_SUCCESS; 1630 } 1631 1632 UCHAR 1633 WINAPI 1634 demGetPhysicalDriveType(IN UCHAR DriveNumber) 1635 { 1636 UNIMPLEMENTED; 1637 return DOSDEVICE_DRIVE_UNKNOWN; 1638 } 1639 1640 BOOL 1641 WINAPI 1642 demIsShortPathName(IN LPCSTR Path, 1643 IN BOOL Unknown) 1644 { 1645 UNIMPLEMENTED; 1646 return FALSE; 1647 } 1648 1649 DWORD 1650 WINAPI 1651 demSetCurrentDirectoryGetDrive(IN LPCSTR CurrentDirectory, 1652 OUT PUCHAR DriveNumber) 1653 { 1654 UNIMPLEMENTED; 1655 return ERROR_SUCCESS; 1656 } 1657 1658 /* EOF */ 1659