Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1895 lines
51 KiB

  1. #include "precomp.h"
  2. #pragma hdrstop
  3. #include <bootfat.h>
  4. #include <bootf32.h>
  5. #include <boot98f.h>
  6. #include <boot98f2.h>
  7. #include <patchbc.h>
  8. //
  9. // Define name of file we use to contain the auxiliary boot sector.
  10. //
  11. #define AUX_BOOT_SECTOR_NAME_A "BOOTSECT.DAT"
  12. #define AUX_BOOT_SECTOR_NAME_W L"BOOTSECT.DAT"
  13. #ifdef UNICODE
  14. #define AUX_BOOT_SECTOR_NAME AUX_BOOT_SECTOR_NAME_W
  15. #else
  16. #define AUX_BOOT_SECTOR_NAME AUX_BOOT_SECTOR_NAME_A
  17. #endif
  18. BOOL CleanUpBootCode;
  19. DWORD CleanUpBootIni;
  20. BOOL
  21. HandleBootFilesWorker_NEC98(
  22. IN TCHAR *SourceDir,
  23. IN TCHAR *DestDir,
  24. IN PTSTR File,
  25. IN BOOL Flag
  26. );
  27. LONG
  28. CalcHiddenSector95(
  29. IN TCHAR DriveLetter
  30. );
  31. BOOL
  32. LoadBootIniString(
  33. IN HINSTANCE ModuleHandle,
  34. IN DWORD MsgId,
  35. OUT PSTR Buffer,
  36. IN DWORD Size
  37. );
  38. //
  39. //
  40. BOOL
  41. CheckSysPartAndReadBootCode(
  42. IN HWND ParentWindow,
  43. OUT WINNT32_SYSPART_FILESYSTEM *Filesystem,
  44. OUT BYTE BootCode[WINNT32_MAX_BOOT_SIZE],
  45. OUT PUINT BootCodeSectorCount
  46. )
  47. /*++
  48. Routine Description:
  49. This routine does some inspection on the x86 system partition
  50. to determine its filesystem and sector size. We only support
  51. 512-byte sectors, and there are code depedencies all over the place
  52. based on this.
  53. If the sector size is wrong or there's a filesystem we don't recognize
  54. then the user is informed.
  55. Arguments:
  56. ParentWindow - supplies window handle of window to be used as
  57. parent/owner in case this routine puts up UI.
  58. Filesystem - if successful, receives the filesystem of the system partition.
  59. BootCode - if successful, receives a copy of the boot code currently
  60. on the disk.
  61. BootCodeSectorCount - if successful, receives the size in 512-byte sectors
  62. of the boot code area for the filesystem on the system partition.
  63. Return Value:
  64. Boolean value indicating whether the system partition is acceptable.
  65. If not, the user will have been informed as to why.
  66. --*/
  67. {
  68. TCHAR DrivePath[4];
  69. DWORD DontCare;
  70. DWORD SectorSize;
  71. TCHAR NameBuffer[100];
  72. BOOL b;
  73. //
  74. // Form root path
  75. //
  76. DrivePath[0] = SystemPartitionDriveLetter;
  77. DrivePath[1] = TEXT(':');
  78. DrivePath[2] = TEXT('\\');
  79. DrivePath[3] = 0;
  80. //
  81. // Check sector size
  82. //
  83. if(!GetDiskFreeSpace(DrivePath,&DontCare,&SectorSize,&DontCare,&DontCare)
  84. || (SectorSize != WINNT32_SECTOR_SIZE)) {
  85. if (!(IsNEC98() && (SectorSize > WINNT32_SECTOR_SIZE))) {
  86. MessageBoxFromMessage(
  87. ParentWindow,
  88. MSG_UNSUPPORTED_SECTOR_SIZE,
  89. FALSE,
  90. AppTitleStringId,
  91. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  92. SystemPartitionDriveLetter
  93. );
  94. return(FALSE);
  95. }
  96. }
  97. //
  98. // Determine file system.
  99. //
  100. b = GetVolumeInformation(
  101. DrivePath,
  102. NULL,0, // don't care about volume name
  103. NULL, // ...or serial #
  104. &DontCare, // ...or max component length
  105. &DontCare, // ... or flags
  106. NameBuffer,
  107. sizeof(NameBuffer)/sizeof(TCHAR)
  108. );
  109. if(!b) {
  110. MessageBoxFromMessage(
  111. ParentWindow,
  112. MSG_UNKNOWN_FS,
  113. FALSE,
  114. AppTitleStringId,
  115. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  116. SystemPartitionDriveLetter
  117. );
  118. return(FALSE);
  119. }
  120. if(!lstrcmpi(NameBuffer,TEXT("NTFS"))) {
  121. *Filesystem = Winnt32FsNtfs;
  122. *BootCodeSectorCount = WINNT32_NTFS_BOOT_SECTOR_COUNT;
  123. b = ReadDiskSectors(
  124. SystemPartitionDriveLetter,
  125. 0,
  126. WINNT32_NTFS_BOOT_SECTOR_COUNT,
  127. WINNT32_SECTOR_SIZE,
  128. BootCode
  129. );
  130. if(!b) {
  131. MessageBoxFromMessage(
  132. ParentWindow,
  133. MSG_DASD_ACCESS_FAILURE,
  134. FALSE,
  135. AppTitleStringId,
  136. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  137. SystemPartitionDriveLetter
  138. );
  139. return(FALSE);
  140. }
  141. } else {
  142. if(!lstrcmpi(NameBuffer,TEXT("FAT")) || !lstrcmpi(NameBuffer,TEXT("FAT32"))) {
  143. //
  144. // Read 1 sector.
  145. //
  146. b = ReadDiskSectors(
  147. SystemPartitionDriveLetter,
  148. 0,
  149. WINNT32_FAT_BOOT_SECTOR_COUNT,
  150. WINNT32_SECTOR_SIZE,
  151. BootCode
  152. );
  153. if(!b) {
  154. MessageBoxFromMessage(
  155. ParentWindow,
  156. MSG_DASD_ACCESS_FAILURE,
  157. FALSE,
  158. AppTitleStringId,
  159. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  160. SystemPartitionDriveLetter
  161. );
  162. return(FALSE);
  163. }
  164. *Filesystem = NameBuffer[3] ? Winnt32FsFat32 : Winnt32FsFat;
  165. *BootCodeSectorCount = WINNT32_FAT_BOOT_SECTOR_COUNT;
  166. } else {
  167. MessageBoxFromMessage(
  168. ParentWindow,
  169. MSG_UNKNOWN_FS,
  170. FALSE,
  171. AppTitleStringId,
  172. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  173. SystemPartitionDriveLetter
  174. );
  175. return(FALSE);
  176. }
  177. }
  178. return(TRUE);
  179. }
  180. BOOL
  181. IsNtBootCode(
  182. IN WINNT32_SYSPART_FILESYSTEM Filesystem,
  183. IN LPBYTE BootCode
  184. )
  185. /*++
  186. Routine Description:
  187. Determine if boot code is for NT by examining the filesystem and
  188. the code itself, looking for the NTLDR string that must be present
  189. in all NT boot code.
  190. If the filesystem is NTFS then it's NT boot code.
  191. If not then we scan backwards in the boot sector looking for the
  192. NTLDR string.
  193. Arguments:
  194. Filesystem - supplies the filesystem on the drive.
  195. BootCode - supplies the boot code read from the drive. Only the first
  196. sector (512 bytes) are examined.
  197. Return Value:
  198. Boolean value indicating whether the boot code is for NT.
  199. There is no error return.
  200. --*/
  201. {
  202. UINT i;
  203. //
  204. // Because the last 2 bytes are the 55aa signature we can
  205. // skip them in the scan.
  206. //
  207. if(Filesystem == Winnt32FsNtfs) {
  208. return(TRUE);
  209. }
  210. for(i=WINNT32_SECTOR_SIZE-7; i>62; --i) {
  211. if(!memcmp("NTLDR",BootCode+i,5)) {
  212. return(TRUE);
  213. }
  214. }
  215. return(FALSE);
  216. }
  217. BOOL
  218. __inline
  219. WriteToBootIni(
  220. IN HANDLE Handle,
  221. IN PCHAR Line
  222. )
  223. {
  224. DWORD bw,l;
  225. l = lstrlenA(Line);
  226. return(WriteFile(Handle,Line,l,&bw,NULL) && (bw == l));
  227. }
  228. BOOL
  229. MungeBootIni(
  230. IN HWND ParentWindow,
  231. IN BOOL SetPreviousOs
  232. )
  233. {
  234. TCHAR BootIniName[16];
  235. TCHAR BootIniBackup[16];
  236. UCHAR BootSectorImageSpec[29];
  237. CHAR HeadlessRedirectSwitches[160];
  238. TCHAR ParamsFile[MAX_PATH];
  239. HANDLE h;
  240. DWORD BootIniSize;
  241. PUCHAR Buffer;
  242. PTCHAR DebugLogBuffer=NULL;
  243. DWORD BytesRead = 0;
  244. BOOL b;
  245. PUCHAR p,next;
  246. BOOL InOsSection;
  247. CHAR c;
  248. CHAR Text[256];
  249. DWORD OldAttributes;
  250. DWORD d;
  251. BOOL UpgradeOSPresent = FALSE;
  252. DWORD attribs;
  253. PUCHAR DefSwitches;
  254. PUCHAR DefSwEnd;
  255. UCHAR temp;
  256. //
  257. // Determine the size of boot.ini, allocate a buffer,
  258. // and read it in. If it isn't there then it will be created.
  259. //
  260. wsprintf(BootIniName,TEXT("%c:\\BOOT.INI"),SystemPartitionDriveLetter);
  261. wsprintf(BootIniBackup,TEXT("%c:\\BOOT.BAK"),SystemPartitionDriveLetter);
  262. h = CreateFile(BootIniName,GENERIC_READ,FILE_SHARE_READ,NULL,OPEN_EXISTING,0,NULL);
  263. if(h == INVALID_HANDLE_VALUE) {
  264. if ( Upgrade && ISNT() ) {
  265. // This is an error that the setup team should probably want
  266. // to look at. If we got this far, then there was a boot.ini
  267. // during pre-copy (look at InspectFileSystems for proof of
  268. // this), but one is missing after doing the copy.
  269. #ifdef PRERELEASE
  270. MessageBox(
  271. ParentWindow,
  272. TEXT("You have encountered a problem the setup team would like to look at.\n\nYou are missing a boot.ini file after the copy step (during MungeBootIni), but there was one much earlier in setup. Something happened between then and now that the setup team (mailto:setuphot) would like to know about."),
  273. TEXT("Winnt32"),
  274. MB_OK | MB_ICONERROR | MB_TASKMODAL);
  275. #else
  276. MessageBoxFromMessage(
  277. ParentWindow,
  278. MSG_UPGRADE_BOOT_INI_MUNGE_MISSING_BOOT_INI,
  279. FALSE,
  280. AppTitleStringId,
  281. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  282. BootIniName);
  283. #endif
  284. b = FALSE;
  285. d = GetLastError();
  286. goto c0;
  287. }
  288. //
  289. // Assume the file does not exist. Allocate a buffer large enough
  290. // to hold a single terminating nul byte.
  291. //
  292. BootIniSize = 0;
  293. Buffer = MALLOC(1);
  294. if(!Buffer) {
  295. b = FALSE;
  296. d = ERROR_NOT_ENOUGH_MEMORY;
  297. goto c0;
  298. }
  299. } else {
  300. //
  301. // Figure out how big the file is.
  302. // Allocate 3 extra characters for final NUL we'll add to make
  303. // parsing easier, and a cr/lf in case the last line is incomplete.
  304. //
  305. BootIniSize = GetFileSize(h,NULL);
  306. if(BootIniSize == (DWORD)(-1)) {
  307. d = GetLastError();
  308. CloseHandle(h);
  309. b = FALSE;
  310. goto c0;
  311. }
  312. Buffer = MALLOC(BootIniSize+3);
  313. DebugLogBuffer = MALLOC( (BootIniSize+3) * sizeof(TCHAR));
  314. if(!Buffer) {
  315. CloseHandle(h);
  316. b = FALSE;
  317. d = ERROR_NOT_ENOUGH_MEMORY;
  318. goto c0;
  319. }
  320. b = ReadFile(h,Buffer,BootIniSize,&BytesRead,NULL);
  321. d = GetLastError();
  322. CloseHandle(h);
  323. if( b && (BootIniSize != BytesRead) ){
  324. // Code to check if due to certain randomness we don't read everything and as a result
  325. // endup overwriting boot.ini
  326. DebugLog( Winnt32LogError, TEXT("Error: BOOT.INI wasn't read properly expected %1: read %2"), 0, BootIniSize, BytesRead);
  327. b = FALSE;
  328. }
  329. if( b && DebugLogBuffer ){
  330. // Log what we read
  331. #ifdef UNICODE
  332. MultiByteToWideChar(
  333. CP_ACP,
  334. 0,
  335. Buffer,
  336. BootIniSize,
  337. DebugLogBuffer,
  338. BootIniSize
  339. );
  340. #else
  341. memcpy( DebugLogBuffer, Buffer, BootIniSize*sizeof(TCHAR));
  342. #endif
  343. DebugLogBuffer[BootIniSize] = 0;
  344. DebugLog( Winnt32LogInformation, TEXT("BOOT.INI record - \n\n%1"), 0, DebugLogBuffer);
  345. }
  346. if(!b) {
  347. goto c1;
  348. }
  349. }
  350. //
  351. // Make sure the last line is properly terminated, and add a terminating nul
  352. // to make parsing a little easier.
  353. //
  354. if(BootIniSize && (Buffer[BootIniSize-1] != '\n') && (Buffer[BootIniSize-1] != '\r')) {
  355. Buffer[BootIniSize++] = '\r';
  356. Buffer[BootIniSize++] = '\n';
  357. }
  358. Buffer[BootIniSize] = 0;
  359. //
  360. // Truncate at control-z if any.
  361. //
  362. if(p = strchr(Buffer,26)) {
  363. if((p > Buffer) && (*(p - 1) != '\n') && (*(p - 1) != '\r')) {
  364. *(p++) = '\r';
  365. *(p++) = '\n';
  366. }
  367. *p = 0;
  368. BootIniSize = p - Buffer;
  369. }
  370. //
  371. // Make sure we can write boot.ini, and make a backup copy.
  372. // (We do not procede unless we can make a backup copy.)
  373. // Then recreate boot.ini.
  374. //
  375. OldAttributes = GetFileAttributes(BootIniName);
  376. SetFileAttributes(BootIniBackup,FILE_ATTRIBUTE_NORMAL);
  377. if(OldAttributes == (DWORD)(-1)) {
  378. //
  379. // Boot.ini didn't exist before. Nothing to do.
  380. //
  381. } else {
  382. //
  383. // Make a backup copy.
  384. //
  385. if(CopyFile(BootIniName,BootIniBackup,FALSE)) {
  386. //
  387. // Attributes could be 0 but not -1. Adding 1 thus allows us to
  388. // use non-0 to mean that we have a backup file.
  389. //
  390. CleanUpBootIni = OldAttributes+1;
  391. } else {
  392. d = GetLastError();
  393. b = FALSE;
  394. goto c1;
  395. }
  396. }
  397. SetFileAttributes(BootIniName,FILE_ATTRIBUTE_NORMAL);
  398. h = CreateFile(
  399. BootIniName,
  400. GENERIC_WRITE,
  401. FILE_SHARE_READ,
  402. NULL,
  403. CREATE_ALWAYS,
  404. FILE_ATTRIBUTE_READONLY | FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM,
  405. NULL
  406. );
  407. if(h == INVALID_HANDLE_VALUE) {
  408. d = GetLastError();
  409. b = FALSE;
  410. goto c2;
  411. }
  412. //
  413. // Regardless of the actual drive letter of the system partition,
  414. // the spec in boot.ini is always C:\...
  415. //
  416. wsprintfA(BootSectorImageSpec,"C:\\%hs\\%hs",LOCAL_BOOT_DIR_A,AUX_BOOT_SECTOR_NAME_A);
  417. //
  418. // Scan the Buffer to see if there is a DefSwitches line,
  419. // to move into new boot.ini in the [boot loader] section.
  420. // If no DefSwitches, just point to a null string to be moved.
  421. // Only process boot.ini up to [operating systems].
  422. //
  423. temp = '\0';
  424. DefSwitches = &temp;
  425. DefSwEnd = NULL;
  426. for(p=Buffer; *p && (p < Buffer+BootIniSize - (sizeof("[operating systems]")-1)); p++) {
  427. if(!_strnicmp(p,"DefSwitches",sizeof("DefSwitches")-1)) {
  428. DefSwEnd = strstr(p, "\n");
  429. if(DefSwEnd){
  430. DefSwEnd++;
  431. if(*DefSwEnd == '\r'){
  432. DefSwEnd++;
  433. }
  434. DefSwitches = p;
  435. temp = *DefSwEnd;
  436. *DefSwEnd = '\0';
  437. }
  438. break;
  439. } else {
  440. if(!_strnicmp(p,"[operating systems]",sizeof("[operating systems]")-1)) {
  441. break;
  442. }
  443. }
  444. }
  445. //
  446. // Take care of the headless setings.
  447. //
  448. HeadlessRedirectSwitches[0] = '\0';
  449. if( HeadlessSelection[0] != TEXT('\0') ) {
  450. //
  451. // They told winnt32.exe some specific headless settings.
  452. // Use these.
  453. //
  454. //
  455. // Convert the user's request into ASCII.
  456. //
  457. #ifdef UNICODE
  458. {
  459. CHAR tmp[80];
  460. WideCharToMultiByte( CP_ACP,
  461. 0,
  462. HeadlessSelection,
  463. -1,
  464. tmp,
  465. sizeof(tmp),
  466. NULL,
  467. NULL );
  468. wsprintfA( HeadlessRedirectSwitches,
  469. "redirect=%s\r\n",
  470. tmp );
  471. }
  472. #else
  473. wsprintfA( HeadlessRedirectSwitches,
  474. "redirect=%s\r\n",
  475. HeadlessSelection );
  476. #endif
  477. } else {
  478. //
  479. // They didn't give us any settings, so see if we can pick
  480. // something up from boot.ini
  481. //
  482. //
  483. // Parse through boot.ini, looking for any 'redirect=' lines.
  484. //
  485. for( p = Buffer; *p && (p < Buffer+BootIniSize - (sizeof("redirect=")-1)); p++ ) {
  486. if(!_strnicmp(p,"[Operat",sizeof("[Operat")-1)) {
  487. //
  488. // We're past the [Boot Loader] section. Stop looking.
  489. //
  490. break;
  491. }
  492. if(!_strnicmp(p,"redirect=",sizeof("redirect=")-1)) {
  493. PUCHAR q = p;
  494. UCHAR temp;
  495. while ((*p != '\r') && (*p != '\n') && *p) {
  496. p++;
  497. }
  498. temp = *p;
  499. *p = '\0';
  500. strcpy(HeadlessRedirectSwitches, q);
  501. //
  502. // We want to make sure that this setting gets put into
  503. // the unattend file too so that textmode will redirect.
  504. // We need to set the global 'HeadlessSelection' so that
  505. // he will get written to winnt.sif after this block.
  506. //
  507. #ifdef UNICODE
  508. MultiByteToWideChar( CP_ACP,
  509. MB_ERR_INVALID_CHARS,
  510. strchr(HeadlessRedirectSwitches, '=')+1,
  511. -1,
  512. HeadlessSelection,
  513. MAX_PATH );
  514. #else
  515. strcpy( HeadlessSelection, strchr(HeadlessRedirectSwitches, '=')+1 );
  516. #endif
  517. strcat(HeadlessRedirectSwitches, "\r\n" );
  518. *p = temp;
  519. }
  520. }
  521. }
  522. //
  523. // Now take care of the 'redirectbaudrate=X' setting.
  524. //
  525. if( HeadlessRedirectSwitches[0] != TEXT('\0') ) {
  526. //
  527. // We got got a direction to redirect. Now see about
  528. // the baudrate.
  529. //
  530. if( HeadlessBaudRate != 0 ) {
  531. CHAR MyHeadlessRedirectBaudRateLine[80] = {0};
  532. wsprintfA( MyHeadlessRedirectBaudRateLine,
  533. "redirectbaudrate=%d\r\n",
  534. HeadlessBaudRate );
  535. strcat( HeadlessRedirectSwitches, MyHeadlessRedirectBaudRateLine );
  536. } else {
  537. //
  538. // They didn't give us any settings, so see if we can pick
  539. // something up from boot.ini
  540. //
  541. //
  542. // Parse through boot.ini, looking for any 'redirectbaudrate=' lines.
  543. //
  544. for( p = Buffer; *p && (p < Buffer+BootIniSize - (sizeof("redirectbaudrate=")-1)); p++ ) {
  545. if(!_strnicmp(p,"[Operat",sizeof("[Operat")-1)) {
  546. //
  547. // We're past the [Boot Loader] section. Stop looking.
  548. //
  549. break;
  550. }
  551. if(!_strnicmp(p,"redirectbaudrate=",sizeof("redirectbaudrate=")-1)) {
  552. PUCHAR q = p;
  553. UCHAR temp;
  554. while ((*p != '\r') && (*p != '\n') && *p) {
  555. p++;
  556. }
  557. temp = *p;
  558. *p = '\0';
  559. strcat(HeadlessRedirectSwitches, q);
  560. strcat(HeadlessRedirectSwitches, "\r\n" );
  561. *p = temp;
  562. //
  563. // Now set the global HeadlessBaudRate variable so
  564. // we'll know what to write in winnt.sif when the time
  565. // comes.
  566. //
  567. p = strchr( q, '=' );
  568. if( p ) {
  569. p++;
  570. HeadlessBaudRate = atoi( p );
  571. }
  572. }
  573. }
  574. }
  575. }
  576. //
  577. // Now generate the name of the parameters file
  578. // and write our headless settings out.
  579. //
  580. BuildSystemPartitionPathToFile( LOCAL_BOOT_DIR,
  581. ParamsFile,
  582. MAX_PATH );
  583. ConcatenatePaths(ParamsFile,WINNT_SIF_FILE,MAX_PATH);
  584. WriteHeadlessParameters( ParamsFile );
  585. wsprintfA(
  586. Text,
  587. "[Boot Loader]\r\nTimeout=5\r\nDefault=%hs\r\n%hs[Operating Systems]\r\n",
  588. BootSectorImageSpec,
  589. HeadlessRedirectSwitches
  590. );
  591. //
  592. // If there were DefSwitches, set the Buffer back to original state
  593. //
  594. if(DefSwEnd){
  595. *DefSwEnd = temp;
  596. }
  597. if(!WriteToBootIni(h,Text)) {
  598. d = GetLastError();
  599. b = FALSE;
  600. DebugLog( Winnt32LogError, TEXT("Error: BOOT.INI wasn't written to properly : LastError - %1"), 0, d);
  601. goto c3;
  602. }
  603. //
  604. // Process each line in boot.ini.
  605. // If it's the setup boot sector line, we'll throw it out.
  606. // For comparison with lines in boot.ini, the drive letter
  607. // is always C even if the system partition is not actually C:.
  608. //
  609. InOsSection = FALSE;
  610. b = TRUE;
  611. for(p=Buffer; *p && b; p=next) {
  612. while((*p==' ') || (*p=='\t')) {
  613. p++;
  614. }
  615. if(*p) {
  616. //
  617. // Find first byte of next line.
  618. //
  619. for(next=p; *next && (*next++ != '\n'); );
  620. //
  621. // Look for start of [operating systems] section
  622. // or at each line in that section.
  623. //
  624. if(InOsSection) {
  625. switch(*p) {
  626. case '[': // end of section.
  627. *p=0; // force break out of loop
  628. break;
  629. case 'C':
  630. case 'c': // potential start of c:\ line
  631. //
  632. // See if it's a line for setup boot.
  633. // If so, ignore it.
  634. //
  635. if(!_strnicmp(p,BootSectorImageSpec,lstrlenA(BootSectorImageSpec))) {
  636. break;
  637. }
  638. //
  639. // If we're supposed to set the previous OS and this is
  640. // a line for the previous OS, ignore it.
  641. //
  642. if(SetPreviousOs && (p[1] == ':') && (p[2] == '\\')
  643. && ((p[3] == '=') || (p[3] == ' ') || (p[3] == '\t'))) {
  644. break;
  645. }
  646. //
  647. // Not a special line, FALL THROUGH to write it out as-is.
  648. //
  649. default:
  650. //
  651. // Random line. write it out.
  652. //
  653. if( Upgrade && ISNT() ){
  654. //
  655. //Check to make sure that in the NT upgrade case we atleast have one valid line
  656. //Using 4 chars as the check as at minimum a valid line should have x=y<CRLF>
  657. //
  658. if( (next - p ) > 4 )
  659. UpgradeOSPresent = TRUE;
  660. }
  661. c = *next;
  662. *next = 0;
  663. b = WriteToBootIni(h,p);
  664. *next = c;
  665. break;
  666. }
  667. } else {
  668. if(!_strnicmp(p,"[operating systems]",19)) {
  669. InOsSection = TRUE;
  670. }
  671. }
  672. }
  673. }
  674. if( ISNT() && Upgrade && !UpgradeOSPresent ){
  675. #ifdef PRERELEASE
  676. //On internal builds we want setuphot to be informed
  677. //when we encounter this
  678. MessageBox(
  679. ParentWindow,
  680. TEXT("You have encountered an error the Setup Team needs to investigate. Send email to SetupHot.(Boot Ini Error)"),
  681. TEXT("Winnt32"),
  682. MB_OK | MB_ICONERROR | MB_TASKMODAL
  683. );
  684. #endif
  685. b = FALSE;
  686. SetLastError( ERROR_INVALID_PARAMETER );
  687. }
  688. //
  689. // Write out our line.
  690. //
  691. if(b) {
  692. CHAR *AnsiStrs[] = {
  693. "Microsoft Windows XP Professional Setup",
  694. "Microsoft Windows 2002 Server Setup",
  695. "Microsoft Windows 2002 Advanced Server Setup",
  696. "Microsoft Windows 2002 Datacenter Server Setup",
  697. "Microsoft Windows 2002 Blade Setup",
  698. "Microsoft Windows 2002 Small Business Server Setup",
  699. "Microsoft Windows XP Setup"
  700. };
  701. DWORD Index = -1;
  702. if (!LoadBootIniString(hInst, AppTitleStringId, Text, sizeof(Text))) {
  703. switch (AppTitleStringId) {
  704. case IDS_APPTITLE_WKS:
  705. Index = 0;
  706. break;
  707. case IDS_APPTITLE_SRV:
  708. Index = 1;
  709. break;
  710. case IDS_APPTITLE_ASRV:
  711. Index = 2;
  712. break;
  713. case IDS_APPTITLE_DAT:
  714. Index = 3;
  715. case IDS_APPTITLE_BLADE:
  716. Index = 4;
  717. case IDS_APPTITLE_SBS:
  718. Index = 5;
  719. default:
  720. Index = 6;
  721. break;
  722. }
  723. strcpy(Text, AnsiStrs[Index]);
  724. }
  725. if((b=WriteToBootIni(h,BootSectorImageSpec))
  726. && (b=WriteToBootIni(h,"=\""))
  727. && (b=WriteToBootIni(h,Text))) {
  728. b = WriteToBootIni(h,"\"\r\n");
  729. }else{
  730. DebugLog( Winnt32LogError, TEXT("Error: Textmode line was not written properly to BOOT.INI"), 0);
  731. }
  732. }
  733. //
  734. // Write out previous OS line if directed to do so.
  735. //
  736. if(b && SetPreviousOs) {
  737. if(b = WriteToBootIni(h,"C:\\=\"")) {
  738. LoadStringA(hInst, Upgrade ? IDS_CANCEL_SETUP:IDS_MICROSOFT_WINDOWS,Text,sizeof(Text));
  739. if(b = WriteToBootIni(h,Text)) {
  740. b = WriteToBootIni(h,"\"\r\n");
  741. }
  742. }
  743. }
  744. if(!b) {
  745. d = GetLastError();
  746. goto c3;
  747. }
  748. d = NO_ERROR;
  749. c3:
  750. CloseHandle(h);
  751. c2:
  752. //
  753. // Restore boot.ini.
  754. //
  755. if(!b && (OldAttributes != (DWORD)(-1))) {
  756. SetFileAttributes(BootIniName,FILE_ATTRIBUTE_NORMAL);
  757. CopyFile(BootIniBackup,BootIniName,FALSE);
  758. SetFileAttributes(BootIniName,OldAttributes);
  759. SetFileAttributes(BootIniBackup,FILE_ATTRIBUTE_NORMAL);
  760. DeleteFile(BootIniBackup);
  761. DebugLog( Winnt32LogError, TEXT("Error processing boot.ini and restored"), 0);
  762. }
  763. c1:
  764. FREE(Buffer);
  765. c0:
  766. if(!b) {
  767. MessageBoxFromMessageAndSystemError(
  768. ParentWindow,
  769. MSG_BOOT_FILE_ERROR,
  770. d,
  771. AppTitleStringId,
  772. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  773. BootIniName
  774. );
  775. }
  776. return(b);
  777. }
  778. VOID
  779. MigrateBootIniData(
  780. VOID
  781. )
  782. {
  783. TCHAR BootIniName[16];
  784. //
  785. // Determine the size of boot.ini, allocate a buffer,
  786. // and read it in. If it isn't there then it will be created.
  787. //
  788. wsprintf(BootIniName,TEXT("%c:\\BOOT.INI"),SystemPartitionDriveLetter);
  789. GetPrivateProfileString(
  790. TEXT("Boot Loader"),
  791. TEXT("Timeout"),
  792. TEXT(""),
  793. Timeout,
  794. sizeof(Timeout)/sizeof(TCHAR),
  795. BootIniName);
  796. }
  797. BOOL
  798. LayNtBootCode(
  799. IN HWND ParentWindow,
  800. IN WINNT32_SYSPART_FILESYSTEM Filesystem,
  801. IN OUT LPBYTE BootCode
  802. )
  803. /*++
  804. Routine Description:
  805. Copy existing boot sector into bootsect.dos and lay down NT boot code.
  806. THIS ROUTINE DOES NOT CHECK THE EXISTING BOOT CODE. The caller must
  807. do that, and not call this routine if the existing boot code is
  808. already for NT. This routine should never be called for an NTFS drive
  809. since by definition that's NT boot code.
  810. Arguments:
  811. ParentWindow - supplies window handle of window to be used as
  812. owner/parent in case this routine puts up UI.
  813. Filesystem - supplies filesystem for system partition, as determined
  814. earlier by CheckSysPartAndReadBootCode(). Either Fat or Fat32.
  815. BootCode - on input, supplies copy of existing boot code read from
  816. the drive. On output, receives copy of new boot code that was
  817. was written to the drive.
  818. Return Value:
  819. Boolean value indicating outcome. If FALSE then the user will have been
  820. informed as to why.
  821. --*/
  822. {
  823. UINT i;
  824. HANDLE h;
  825. TCHAR FileName[] = TEXT("?:\\BOOTSECT.DOS");
  826. DWORD d;
  827. BOOL b;
  828. //
  829. // Nt 3.51 will bugcheck here if they have an adaptec
  830. // 2940 card. Return if we're on 3.51. Note that
  831. // if any of the APIs fail, or anything goes wrong
  832. // in here, we just continue, assuming we're not
  833. // on NT 3.51.
  834. //
  835. if(!IsNEC98() && ISNT() && (BuildNumber <= NT351)) {
  836. return TRUE;
  837. }
  838. //
  839. // We may want to update the boot sector even if it
  840. // is NT boot code. In that case, we don't want to
  841. // go blast out a new bootsect.dos. Check first.
  842. //
  843. // If this process is called during /cmdcons,
  844. // the BOOTSECT.DOS should not be created on NEC98
  845. //
  846. if((IsNEC98() && !(BuildCmdcons)) || !(ISNT() || IsNtBootCode(Filesystem,BootCode)) ) {
  847. //
  848. // Write out existing boot sector to bootsect.dos.
  849. // We only move a single sector, which is correct in Fat
  850. // and Fat32 cases. The NT Fat32 boot code looks in sector
  851. // 12 for its second sector, so no special casing is required.
  852. //
  853. FileName[0] = SystemPartitionDriveLetter;
  854. SetFileAttributes(FileName,FILE_ATTRIBUTE_NORMAL);
  855. h = CreateFile(
  856. FileName,
  857. GENERIC_WRITE,
  858. FILE_SHARE_READ,
  859. NULL,
  860. CREATE_ALWAYS,
  861. FILE_ATTRIBUTE_READONLY | FILE_ATTRIBUTE_SYSTEM | FILE_FLAG_WRITE_THROUGH | FILE_FLAG_SEQUENTIAL_SCAN,
  862. NULL
  863. );
  864. if(h == INVALID_HANDLE_VALUE) {
  865. MessageBoxFromMessageAndSystemError(
  866. ParentWindow,
  867. MSG_BOOT_FILE_ERROR,
  868. GetLastError(),
  869. AppTitleStringId,
  870. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  871. FileName
  872. );
  873. return(FALSE);
  874. }
  875. b = WriteFile(h,BootCode,WINNT32_SECTOR_SIZE,&d,NULL);
  876. d = GetLastError();
  877. CloseHandle(h);
  878. if(!b) {
  879. MessageBoxFromMessageAndSystemError(
  880. ParentWindow,
  881. MSG_BOOT_FILE_ERROR,
  882. d,
  883. AppTitleStringId,
  884. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  885. FileName
  886. );
  887. return(FALSE);
  888. }
  889. }
  890. //
  891. // While upgrading Win9X with FAT32 system partition installations,
  892. // update the BPB's heads value to reflect the actual value
  893. //
  894. if (!ISNT() && (Filesystem == Winnt32FsFat32)) {
  895. if (!PatchBootCode(Filesystem,
  896. SystemPartitionDriveLetter,
  897. (PUCHAR)BootCode,
  898. sizeof(Fat32BootCode))) {
  899. //
  900. // for failure case just log a winnt32.log error message
  901. // indicating the error
  902. //
  903. DebugLog(Winnt32LogError,
  904. TEXT("Could not update the FAT32 system partition's boot sector's\r\n")
  905. TEXT(" Bios Parameter Block's heads value"),
  906. 0);
  907. }
  908. }
  909. //
  910. // Now lay the NT code itself down onto the disk. We copy the non-BPB parts
  911. // of the appropriate template code into the caller's bootcode buffer.
  912. // Take advantage of the offset part of the jump instruction at the start
  913. // of the boot code (like eb 3c 90) to tell us where the BPB ends and
  914. // the code begins.
  915. //
  916. switch(Filesystem) {
  917. case Winnt32FsFat:
  918. {
  919. BYTE BootCodeBuffer[WINNT32_MAX_BOOT_SIZE];
  920. if (IsNEC98())
  921. {
  922. CopyMemory(BootCodeBuffer,PC98FatBootCode,sizeof(PC98FatBootCode));
  923. // NEC98 need set to HiddenSector(Bpb Index 0x011) value in BPB.
  924. // Hiddensector value is how many sectors from sector 0
  925. // This spec is NEC98 only.
  926. *(LONG *)&BootCodeBuffer[0x011 + 11]
  927. = CalcHiddenSector(SystemPartitionDriveLetter,
  928. *(SHORT *)&BootCodeBuffer[11]);
  929. } else {
  930. CopyMemory(BootCodeBuffer,FatBootCode,sizeof(FatBootCode));
  931. }
  932. CopyMemory(BootCode,BootCodeBuffer,3);
  933. CopyMemory(
  934. BootCode + BootCodeBuffer[1] + 2,
  935. BootCodeBuffer + BootCodeBuffer[1] + 2,
  936. WINNT32_SECTOR_SIZE - (BootCodeBuffer[1] + 2)
  937. );
  938. }
  939. break;
  940. case Winnt32FsFat32:
  941. //
  942. // In the FAT32 case we also lay down NT's second sector at sector 12.
  943. //
  944. {
  945. BYTE BootCodeBuffer[WINNT32_MAX_BOOT_SIZE];
  946. if (IsNEC98())
  947. {
  948. CopyMemory(BootCodeBuffer,PC98Fat32BootCode,sizeof(PC98Fat32BootCode));
  949. } else {
  950. CopyMemory(BootCodeBuffer,Fat32BootCode,sizeof(Fat32BootCode));
  951. }
  952. b = WriteDiskSectors( SystemPartitionDriveLetter,
  953. 12,
  954. 1,
  955. WINNT32_SECTOR_SIZE,
  956. BootCodeBuffer+1024 );
  957. if(b) {
  958. CopyMemory(BootCode,BootCodeBuffer,3);
  959. CopyMemory( BootCode + BootCodeBuffer[1] + 2,
  960. BootCodeBuffer + BootCodeBuffer[1] + 2,
  961. WINNT32_SECTOR_SIZE - (BootCodeBuffer[1] + 2) );
  962. }
  963. }
  964. break;
  965. default:
  966. //
  967. // We should never get here.
  968. //
  969. b = FALSE;
  970. break;
  971. }
  972. if(b) {
  973. b = WriteDiskSectors(
  974. SystemPartitionDriveLetter,
  975. 0,
  976. 1,
  977. WINNT32_SECTOR_SIZE,
  978. BootCode
  979. );
  980. if(b) {
  981. CleanUpBootCode = TRUE;
  982. }
  983. }
  984. if(!b) {
  985. MessageBoxFromMessage(
  986. ParentWindow,
  987. MSG_DASD_ACCESS_FAILURE,
  988. FALSE,
  989. AppTitleStringId,
  990. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  991. SystemPartitionDriveLetter
  992. );
  993. }
  994. return(b);
  995. }
  996. BOOL
  997. CreateAuxiliaryBootSector(
  998. IN HWND ParentWindow,
  999. IN WINNT32_SYSPART_FILESYSTEM Filesystem,
  1000. IN LPBYTE BootCode,
  1001. IN UINT BootCodeSectorCount
  1002. )
  1003. /*++
  1004. Routine Description:
  1005. When ntldr sees an entry in boot.ini that starts with the magic text "C:\"
  1006. it will look to see if the item specifies a filename, and if so it will
  1007. assume that the file is a boot sector, load it, and jump to it.
  1008. We place an entry in boot.ini for C:\$WIN_NT$.~BT\BOOTSECT.DAT, and place
  1009. our special boot sector(s) in that file. Our sector is special because it
  1010. loads $LDR$ instead of NTLDR, allowing us to boot into setup without
  1011. disturbing the "standard" ntldr-based boot.
  1012. This routine exmaines the boot code on the disk, changes NTLDR to $LDR$
  1013. and writes the result out to x:\$WIN_NT$.~BT\BOOTSECT.DAT.
  1014. This code assumes a sector size of 512 bytes.
  1015. Arguments:
  1016. ParentWindow - supplies a window handle for a window to act as parent/owner
  1017. for any ui that gets displayed by this routine.
  1018. Filesystem - supplies a value that indicates the filesystem on the
  1019. system partition.
  1020. BootCode - supplies a buffer containing a copy of the boot code that is
  1021. actually on the disk.
  1022. BootCodeSectorCount - supplies the number of sectors the boot code
  1023. occupies on-disk (and thus indicates the size of the BootCode buffer).
  1024. Return Value:
  1025. Boolean value indicating outcome. If FALSE, the user will have been
  1026. informed about why".
  1027. --*/
  1028. {
  1029. UINT i;
  1030. TCHAR NameBuffer[MAX_PATH];
  1031. HANDLE hFile;
  1032. BOOL b;
  1033. DWORD DontCare;
  1034. //
  1035. // Change NTLDR to $LDR$. NTFS stores it in unicode in its boot sector
  1036. // so 2 separate algorithms are needed.
  1037. //
  1038. if(Filesystem == Winnt32FsNtfs) {
  1039. for(i=1014; i>62; i-=2) {
  1040. if(!memcmp("N\000T\000L\000D\000R\000",BootCode+i,10)) {
  1041. //
  1042. // Do NOT use _lstrcpynW here since there is no
  1043. // way to get it to do the right thing without overwriting
  1044. // the word after $LDR$ with a terminating 0. Doing that
  1045. // breaks boot.
  1046. //
  1047. CopyMemory(BootCode+i,AUX_BS_NAME_W,10);
  1048. break;
  1049. }
  1050. }
  1051. } else {
  1052. for(i=505; i>62; --i) {
  1053. //
  1054. // Scan for full name with spaces so we don't find a boot message
  1055. // by accident.
  1056. //
  1057. if(!memcmp("NTLDR ",BootCode+i,11)) {
  1058. strncpy(BootCode+i,AUX_BS_NAME_A,5);
  1059. break;
  1060. }
  1061. }
  1062. }
  1063. //
  1064. // Form name of boot sector image file.
  1065. //
  1066. wsprintf(
  1067. NameBuffer,
  1068. TEXT("%c:\\%s\\%s"),
  1069. SystemPartitionDriveLetter,
  1070. LOCAL_BOOT_DIR,
  1071. AUX_BOOT_SECTOR_NAME
  1072. );
  1073. //
  1074. // Write boot sector image into file.
  1075. //
  1076. SetFileAttributes(NameBuffer,FILE_ATTRIBUTE_NORMAL);
  1077. hFile = CreateFile(NameBuffer,GENERIC_WRITE,0,NULL,CREATE_ALWAYS,0,NULL);
  1078. if(hFile == INVALID_HANDLE_VALUE) {
  1079. MessageBoxFromMessageAndSystemError(
  1080. ParentWindow,
  1081. MSG_BOOT_FILE_ERROR,
  1082. GetLastError(),
  1083. AppTitleStringId,
  1084. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  1085. NameBuffer
  1086. );
  1087. return(FALSE);
  1088. }
  1089. //
  1090. // We have a timing bug that we're going to workaround for the
  1091. // time being...
  1092. //
  1093. i = 0;
  1094. b = FALSE;
  1095. while( (i < 10) && (b == FALSE) ) {
  1096. Sleep( 500 );
  1097. b = WriteFile(hFile,BootCode,BootCodeSectorCount*WINNT32_SECTOR_SIZE,&DontCare,NULL);
  1098. if( !b ) {
  1099. DontCare = GetLastError();
  1100. }
  1101. i++;
  1102. }
  1103. if(!b) {
  1104. MessageBoxFromMessageAndSystemError(
  1105. ParentWindow,
  1106. MSG_BOOT_FILE_ERROR,
  1107. DontCare,
  1108. AppTitleStringId,
  1109. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  1110. NameBuffer
  1111. );
  1112. //
  1113. // Set this back before we ship Beta2!
  1114. // -matth
  1115. //
  1116. #if 1
  1117. //
  1118. // Now try again.
  1119. //
  1120. b = WriteFile(hFile,BootCode,BootCodeSectorCount*WINNT32_SECTOR_SIZE,&DontCare,NULL);
  1121. #endif
  1122. }
  1123. CloseHandle(hFile);
  1124. //
  1125. // Success if we get here.
  1126. //
  1127. return(b);
  1128. }
  1129. BOOL
  1130. DoX86BootStuff(
  1131. IN HWND ParentWindow
  1132. )
  1133. {
  1134. WINNT32_SYSPART_FILESYSTEM Filesystem;
  1135. BYTE BootCode[WINNT32_MAX_BOOT_SIZE];
  1136. UINT BootCodeSectorCount;
  1137. BOOL AlreadyNtBoot;
  1138. TCHAR Filename[13];
  1139. WIN32_FIND_DATA FindData;
  1140. HANDLE FindHandle;
  1141. BOOL b;
  1142. //
  1143. // On Win95, make sure we have NTLDR on the system partition,
  1144. // otherwise it makes no sense to lay NT boot code. This is
  1145. // a robustness thing to catch the case where an error occurred
  1146. // copying that file and the user skipped, etc. Otherwise we could
  1147. // end up getting the user into a situation where he can't boot.
  1148. //
  1149. if(!ISNT()) {
  1150. wsprintf(Filename,TEXT("%c:\\NTLDR"),SystemPartitionDriveLetter);
  1151. FindHandle = FindFirstFile(Filename,&FindData);
  1152. if(FindHandle == INVALID_HANDLE_VALUE) {
  1153. b = FALSE;
  1154. } else {
  1155. FindClose(FindHandle);
  1156. if((FindData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) || !FindData.nFileSizeLow) {
  1157. b = FALSE;
  1158. } else {
  1159. b = TRUE;
  1160. }
  1161. }
  1162. if(!b) {
  1163. MessageBoxFromMessage(
  1164. ParentWindow,
  1165. MSG_NTLDR_NOT_COPIED,
  1166. FALSE,
  1167. AppTitleStringId,
  1168. MB_OK | MB_ICONERROR | MB_TASKMODAL,
  1169. SystemPartitionDriveLetter
  1170. );
  1171. return(FALSE);
  1172. }
  1173. }
  1174. //
  1175. // Check out C:. Sector size must be 512 bytes and it has to be
  1176. // formatted in a filesystem we recognize -- FAT, FAT32, or NTFS
  1177. // (NT 3.51 also supported HPFS, but we assume we would not have
  1178. // gotten here if the drive is HPFS).
  1179. //
  1180. if(!CheckSysPartAndReadBootCode(ParentWindow,&Filesystem,BootCode,&BootCodeSectorCount)) {
  1181. return(FALSE);
  1182. }
  1183. //
  1184. // If we're running on Win95 check the existing boot code to see whether
  1185. // it's already for NT. If on NT assume the boot code is correct.
  1186. // That assumption could be bogus in some marginal cases (such as when
  1187. // the user boots from a floppy with ntldr on it and C: is corrupt
  1188. // or has been re-sys'ed, etc), but we ignore these issues.
  1189. //
  1190. AlreadyNtBoot = ISNT() ? TRUE : IsNtBootCode(Filesystem,BootCode);
  1191. //
  1192. // Munge boot.ini. We do this before laying NT boot code. If we did it
  1193. // afterwards and it failed, then the user could have NT boot code but no
  1194. // boot.ini, which would be bad news.
  1195. //
  1196. if(!MungeBootIni(ParentWindow,!AlreadyNtBoot)) {
  1197. return(FALSE);
  1198. }
  1199. //
  1200. // If BOOTSEC.DOS exist, We Need save BOOTSEC.DOS on NEC98 System.
  1201. //Some case, It is different to Now boot sector. It is created by
  1202. //NT4.
  1203. // NEC970725
  1204. // If this process is called during /cmdcons,
  1205. // the BOOTSECT.DOS should not be renamed "BOOTSECT.NEC" on NEC98
  1206. //
  1207. if (IsNEC98() && !(BuildCmdcons)){
  1208. TCHAR FileNameOld[16],FileNameNew[163];
  1209. FileNameOld[0] = FileNameNew[0] = SystemPartitionDriveLetter;
  1210. FileNameOld[1] = FileNameNew[1] = TEXT(':');
  1211. FileNameOld[2] = FileNameNew[2] = TEXT('\\');
  1212. lstrcpy(FileNameOld+3,TEXT("BOOTSECT.DOS"));
  1213. lstrcpy(FileNameNew+3,TEXT("BOOTSECT.NEC"));
  1214. SetFileAttributes(FileNameOld,FILE_ATTRIBUTE_NORMAL);
  1215. DeleteFile(FileNameNew);
  1216. MoveFile(FileNameOld, FileNameNew);
  1217. }
  1218. //
  1219. // If not already NT boot, copy existing boot code into bootsect.dos
  1220. // and lay down NT boot code.
  1221. //
  1222. // We're going to start writing new boot code if we're on anything
  1223. // but an NTFS drive.
  1224. //
  1225. if( (!AlreadyNtBoot) || (Filesystem != Winnt32FsNtfs) ) {
  1226. if( !LayNtBootCode(ParentWindow,Filesystem,BootCode) ) {
  1227. return(FALSE);
  1228. }
  1229. }
  1230. //
  1231. // Create the auxiliary boot code file, which is a copy of the NT
  1232. // boot code for the drive, with NTLDR changed to $LDR$.
  1233. //
  1234. if( (ForcedSystemPartition) &&
  1235. (UserSpecifiedLocalSourceDrive) &&
  1236. (ForcedSystemPartition == UserSpecifiedLocalSourceDrive) ) {
  1237. TCHAR FileNameOld[32],FileNameNew[32];
  1238. //
  1239. // The OEM is making a bootable disk with local source for a
  1240. // preinstall scenario. We can avoid any drive geometry dependence
  1241. // by simply booting the setupldr instead of using the ntldr->
  1242. // bootsect.dat->setupldr. To do this, we'll simply copy setupldr
  1243. // over ntldr. Note that we're removing his ability to boot anything
  1244. // other than textmode setup here, so be aware.
  1245. //
  1246. //
  1247. // Unlock ntldr.
  1248. //
  1249. FileNameOld[0] = FileNameNew[0] = ForcedSystemPartition;
  1250. FileNameOld[1] = FileNameNew[1] = TEXT(':');
  1251. FileNameOld[2] = FileNameNew[2] = TEXT('\\');
  1252. lstrcpy(FileNameOld+3,TEXT("$LDR$"));
  1253. lstrcpy(FileNameNew+3,TEXT("NTLDR"));
  1254. SetFileAttributes(FileNameNew,FILE_ATTRIBUTE_NORMAL);
  1255. //
  1256. // Move $LDR$ to NTLDR
  1257. //
  1258. DeleteFile(FileNameNew);
  1259. MoveFile(FileNameOld, FileNameNew);
  1260. } else {
  1261. if(!CreateAuxiliaryBootSector(ParentWindow,Filesystem,BootCode,BootCodeSectorCount)) {
  1262. return(FALSE);
  1263. }
  1264. }
  1265. return(TRUE);
  1266. }
  1267. BOOL
  1268. RestoreBootSector(
  1269. VOID
  1270. )
  1271. {
  1272. TCHAR Name[MAX_PATH];
  1273. BYTE Buffer[WINNT32_MAX_BOOT_SIZE];
  1274. DWORD BytesRead;
  1275. BOOL b;
  1276. HANDLE h;
  1277. //
  1278. // If we didn't get to the point of writing new boot code,
  1279. // then there's nothing to do.
  1280. //
  1281. if(!CleanUpBootCode) {
  1282. return(TRUE);
  1283. }
  1284. //
  1285. // Try to put bootsect.dos back onto the boot sector.
  1286. //
  1287. wsprintf(
  1288. Name,
  1289. TEXT("%c:\\%s\\BOOTSECT.DOS"),
  1290. SystemPartitionDriveLetter,
  1291. LOCAL_BOOT_DIR
  1292. );
  1293. h = CreateFile(Name,GENERIC_READ,0,NULL,OPEN_EXISTING,FILE_FLAG_SEQUENTIAL_SCAN,NULL);
  1294. if(h == INVALID_HANDLE_VALUE) {
  1295. b = FALSE;
  1296. } else {
  1297. b = ReadFile(h,Buffer,WINNT32_SECTOR_SIZE,&BytesRead,NULL);
  1298. CloseHandle(h);
  1299. if(b) {
  1300. b = WriteDiskSectors(
  1301. SystemPartitionDriveLetter,
  1302. 0,
  1303. 1,
  1304. WINNT32_SECTOR_SIZE,
  1305. Buffer
  1306. );
  1307. if(b) {
  1308. //
  1309. // If this worked then we don't need ntldr, ntdetect.com, or boot.ini.
  1310. // If is possible that these files were there already before
  1311. // and we're thus "overcleaning" but we shouldn't get here
  1312. // unless we overwrote non-nt boot code with nt boot code.
  1313. // Thus putting back bootsect.dos restores non-NT boot code,
  1314. // so this shouldn't be too destructive.
  1315. //
  1316. Name[0] = SystemPartitionDriveLetter;
  1317. Name[1] = TEXT(':');
  1318. Name[2] = TEXT('\\');
  1319. lstrcpy(Name+3,TEXT("NTLDR"));
  1320. SetFileAttributes(Name,FILE_ATTRIBUTE_NORMAL);
  1321. DeleteFile(Name);
  1322. lstrcpy(Name+3,TEXT("NTDETECT.COM"));
  1323. SetFileAttributes(Name,FILE_ATTRIBUTE_NORMAL);
  1324. DeleteFile(Name);
  1325. wsprintf(Name+3,TEXT("BOOT.INI"));
  1326. SetFileAttributes(Name,FILE_ATTRIBUTE_NORMAL);
  1327. DeleteFile(Name);
  1328. }
  1329. }
  1330. }
  1331. return(b);
  1332. }
  1333. BOOL
  1334. RestoreBootIni(
  1335. VOID
  1336. )
  1337. {
  1338. BOOL b = TRUE;
  1339. TCHAR BootIniFile[12] = TEXT("X:\\BOOT.INI");
  1340. TCHAR BackupFile[12] = TEXT("X:\\BOOT.BAK");
  1341. if(CleanUpBootIni) {
  1342. CleanUpBootIni--;
  1343. BootIniFile[0] = SystemPartitionDriveLetter;
  1344. BackupFile[0] = SystemPartitionDriveLetter;
  1345. SetFileAttributes(BootIniFile,FILE_ATTRIBUTE_NORMAL);
  1346. if(CopyFile(BackupFile,BootIniFile,FALSE)) {
  1347. SetFileAttributes(BackupFile,FILE_ATTRIBUTE_NORMAL);
  1348. DeleteFile(BackupFile);
  1349. SetFileAttributes(BootIniFile,CleanUpBootIni);
  1350. } else {
  1351. b = FALSE;
  1352. }
  1353. }
  1354. return(b);
  1355. }
  1356. BOOL
  1357. SaveRestoreBootFiles_NEC98(
  1358. IN UCHAR Flag
  1359. )
  1360. {
  1361. PTSTR BackupFiles[] = { TEXT("\\BOOT.INI"),
  1362. TEXT("\\NTDETECT.COM"),
  1363. TEXT("\\NTLDR"),
  1364. NULL
  1365. };
  1366. PTSTR BackupFiles2[] = { TEXT("\\") AUX_BS_NAME, TEXT("\\") TEXTMODE_INF, NULL };
  1367. UINT i;
  1368. TCHAR SystemDir[3];
  1369. SystemDir[0] = SystemPartitionDriveLetter;
  1370. SystemDir[1] = TEXT(':');
  1371. SystemDir[2] = 0;
  1372. if (Flag == NEC98RESTOREBOOTFILES){
  1373. //
  1374. // Restore boot files.
  1375. //
  1376. for(i=0; BackupFiles[i] ; i++) {
  1377. HandleBootFilesWorker_NEC98(
  1378. LocalBackupDirectory,
  1379. SystemDir,
  1380. BackupFiles[i],
  1381. TRUE
  1382. );
  1383. }
  1384. //
  1385. // Delete tmp files.
  1386. //
  1387. for(i=0; BackupFiles2[i] ; i++) {
  1388. HandleBootFilesWorker_NEC98(
  1389. NULL,
  1390. SystemDir,
  1391. BackupFiles2[i],
  1392. FALSE
  1393. );
  1394. }
  1395. } else {
  1396. if (CreateDirectory(LocalBackupDirectory, NULL))
  1397. for (i = 0; BackupFiles[i] ; i++) {
  1398. HandleBootFilesWorker_NEC98(SystemDir,
  1399. LocalBackupDirectory,
  1400. BackupFiles[i],
  1401. TRUE);
  1402. }
  1403. }
  1404. return(TRUE);
  1405. }
  1406. BOOL
  1407. HandleBootFilesWorker_NEC98(
  1408. IN TCHAR *SourceDir,
  1409. IN TCHAR *DestDir,
  1410. IN PTSTR File,
  1411. IN BOOL Flag
  1412. )
  1413. {
  1414. TCHAR SourceFile[MAX_PATH];
  1415. TCHAR TargetFile[MAX_PATH];
  1416. DWORD OldAttributes;
  1417. if ((!DestDir) || ((!SourceDir)&&Flag)) {
  1418. return(FALSE);
  1419. }
  1420. lstrcpy(TargetFile, DestDir);
  1421. lstrcat(TargetFile, File);
  1422. if (SourceDir) {
  1423. lstrcpy(SourceFile, SourceDir);
  1424. lstrcat(SourceFile, File);
  1425. }
  1426. if (Flag) {
  1427. OldAttributes = GetFileAttributes(TargetFile);
  1428. SetFileAttributes(TargetFile,FILE_ATTRIBUTE_NORMAL);
  1429. if (!CopyFile(SourceFile,TargetFile,FALSE)) {
  1430. Sleep(500);
  1431. if (!CopyFile(SourceFile,TargetFile,FALSE)) {
  1432. return(FALSE);
  1433. }
  1434. }
  1435. if (OldAttributes != (DWORD)(-1)) {
  1436. SetFileAttributes(TargetFile,OldAttributes & ~FILE_ATTRIBUTE_COMPRESSED);
  1437. }
  1438. } else {
  1439. SetFileAttributes(TargetFile,FILE_ATTRIBUTE_NORMAL);
  1440. DeleteFile(TargetFile);
  1441. }
  1442. return(TRUE);
  1443. }
  1444. BOOL
  1445. PatchTextIntoBootCode(
  1446. VOID
  1447. )
  1448. {
  1449. BOOLEAN b;
  1450. CHAR Missing[100];
  1451. CHAR DiskErr[100];
  1452. CHAR PressKey[100];
  1453. if(LoadStringA(hInst,IDS_BOOTMSG_FAT_NTLDR_MISSING,Missing,sizeof(Missing))
  1454. && LoadStringA(hInst,IDS_BOOTMSG_FAT_DISKERROR,DiskErr,sizeof(DiskErr))
  1455. && LoadStringA(hInst,IDS_BOOTMSG_FAT_PRESSKEY,PressKey,sizeof(PressKey))) {
  1456. CharToOemA(Missing,Missing);
  1457. CharToOemA(DiskErr,DiskErr);
  1458. CharToOemA(PressKey,PressKey);
  1459. if(b = PatchMessagesIntoFatBootCode(FatBootCode,FALSE,Missing,DiskErr,PressKey)) {
  1460. b = PatchMessagesIntoFatBootCode(Fat32BootCode,TRUE,Missing,DiskErr,PressKey);
  1461. }
  1462. } else {
  1463. b = FALSE;
  1464. }
  1465. return((BOOL)b);
  1466. }
  1467. LONG
  1468. CalcHiddenSector(
  1469. IN TCHAR DriveLetter,
  1470. IN SHORT Bps
  1471. )
  1472. {
  1473. TCHAR HardDiskName[] = TEXT("\\\\.\\?:");
  1474. HANDLE hDisk;
  1475. PARTITION_INFORMATION partition_info;
  1476. DWORD DataSize;
  1477. if (!ISNT()){
  1478. return(CalcHiddenSector95(DriveLetter));
  1479. } else {
  1480. HardDiskName[4] = DriveLetter;
  1481. hDisk = CreateFileW((const unsigned short *)HardDiskName,
  1482. GENERIC_READ | GENERIC_WRITE,
  1483. FILE_SHARE_READ | FILE_SHARE_WRITE,
  1484. NULL,
  1485. OPEN_EXISTING,
  1486. FILE_ATTRIBUTE_NORMAL,
  1487. NULL
  1488. );
  1489. if(hDisk == INVALID_HANDLE_VALUE) {
  1490. return 0L;
  1491. }
  1492. DeviceIoControl(hDisk,
  1493. IOCTL_DISK_GET_PARTITION_INFO,
  1494. NULL,
  1495. 0,
  1496. &partition_info,
  1497. sizeof(PARTITION_INFORMATION),
  1498. &DataSize,
  1499. NULL);
  1500. CloseHandle(hDisk);
  1501. return(LONG)(partition_info.StartingOffset.QuadPart / Bps);
  1502. }
  1503. }
  1504. LONG
  1505. CalcHiddenSector95(
  1506. IN TCHAR DriveLetter
  1507. )
  1508. {
  1509. #define WINNT_WIN95HLP_GET1STSECTOR_W L"GetFirstSectorNo32"
  1510. #define WINNT_WIN95HLP_GET1STSECTOR_A "GetFirstSectorNo32"
  1511. #define NEC98_DLL_NAME_W L"98PTN32.DLL"
  1512. #define NEC98_DLL_NAME_A "98PTN32.DLL"
  1513. #ifdef UNICODE
  1514. #define WINNT_WIN95HLP_GET1STSECTOR WINNT_WIN95HLP_GET1STSECTOR_W
  1515. #define NEC98_DLL_NAME NEC98_DLL_NAME_W
  1516. #else
  1517. #define WINNT_WIN95HLP_GET1STSECTOR WINNT_WIN95HLP_GET1STSECTOR_A
  1518. #define NEC98_DLL_NAME NEC98_DLL_NAME_A
  1519. #endif
  1520. typedef DWORD (CALLBACK WINNT32_PLUGIN_WIN95_GET1STSECTOR_PROTOTYPE)(int, WORD);
  1521. typedef WINNT32_PLUGIN_WIN95_GET1STSECTOR_PROTOTYPE * PWINNT32_PLUGIN_WIN95_GET1STSECTOR;
  1522. TCHAR ModuleName[MAX_PATH], *p;
  1523. HINSTANCE Pc98ModuleHandle;
  1524. PWINNT32_PLUGIN_WIN95_GET1STSECTOR Get1stSector;
  1525. LONG NumSectors = 0; // indicates failure
  1526. if(!GetModuleFileName (NULL, ModuleName, MAX_PATH) ||
  1527. (!(p=_tcsrchr(ModuleName, TEXT('\\')))) ) {
  1528. return 0;
  1529. }
  1530. *p= 0;
  1531. ConcatenatePaths (ModuleName, NEC98_DLL_NAME, MAX_PATH);
  1532. //
  1533. // Load library
  1534. //
  1535. Pc98ModuleHandle = LoadLibraryEx(
  1536. ModuleName,
  1537. NULL,
  1538. LOAD_WITH_ALTERED_SEARCH_PATH
  1539. );
  1540. if (Pc98ModuleHandle) {
  1541. //
  1542. // Get entry point
  1543. //
  1544. Get1stSector= (PWINNT32_PLUGIN_WIN95_GET1STSECTOR)
  1545. GetProcAddress (Pc98ModuleHandle,
  1546. (const char *)WINNT_WIN95HLP_GET1STSECTOR);
  1547. if (Get1stSector) {
  1548. //
  1549. // the second parameter must be 0.
  1550. // if 0 is returned, it indicates the function failed.
  1551. //
  1552. NumSectors = (LONG)Get1stSector((int)DriveLetter, (WORD)0);
  1553. }
  1554. FreeLibrary(Pc98ModuleHandle);
  1555. }
  1556. return NumSectors;
  1557. }