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.

2203 lines
65 KiB

  1. /*++
  2. Copyright (c) 1992 Microsoft Corporation
  3. Module Name:
  4. Spntupg.c
  5. Abstract:
  6. initializing and maintaining list of nts to upgrade
  7. Author:
  8. Sunil Pai (sunilp) 10-Nov-1993
  9. Revision History:
  10. --*/
  11. #include "spprecmp.h"
  12. #pragma hdrstop
  13. //
  14. // Major/minor version numbers of the system we're upgrading *from*
  15. // if upgrading.
  16. //
  17. ULONG OldMajorVersion,OldMinorVersion;
  18. //**************************************************************
  19. // S E L E C T I N G N T T O U P G R A D E S T U F F
  20. //**************************************************************
  21. #define MENU_LEFT_X 3
  22. #define MENU_WIDTH (VideoVars.ScreenWidth-(2*MENU_LEFT_X))
  23. #define MENU_INDENT 4
  24. VOID
  25. pSpStepUpValidate(
  26. VOID //IN BOOLEAN Server
  27. );
  28. BOOLEAN
  29. SpGetStepUpMode(
  30. IN PWSTR PidExtraData,
  31. BOOLEAN *StepUpMode
  32. );
  33. BOOLEAN
  34. pSpGetCurrentInstallVariation(
  35. IN PWSTR szPid20,
  36. OUT LPDWORD CurrentInstallVariation
  37. );
  38. VOID
  39. SpGetUpgDriveLetter(
  40. IN WCHAR DriveLetter,
  41. IN PWCHAR Buffer,
  42. IN ULONG BufferSize,
  43. IN BOOL AddColon
  44. );
  45. VOID
  46. SpCantFindBuildToUpgrade(
  47. VOID
  48. );
  49. #ifdef _X86_
  50. BOOLEAN
  51. SpIsWin9xMsdosSys(
  52. IN PDISK_REGION Region,
  53. OUT PSTR* Win9xPath
  54. );
  55. #endif
  56. VOID
  57. SpGetFileVersion(
  58. IN PVOID ImageBase,
  59. OUT PULONGLONG Version
  60. );
  61. ENUMUPGRADETYPE
  62. SpFindNtToUpgrade(
  63. IN PVOID SifHandle,
  64. OUT PDISK_REGION *TargetRegion,
  65. OUT PWSTR *TargetPath,
  66. OUT PDISK_REGION *SystemPartitionRegion,
  67. OUT PWSTR *SystemPartitionDirectory
  68. )
  69. /*++
  70. Routine Description:
  71. This goes through the list of NTs on the system and finds out which are
  72. upgradeable. Presents the information to the user and selects if he
  73. wishes to upgrade an installed NT / install a fresh NT into the same
  74. directory / select a different location for Windows NT.
  75. If the chosen target is too full user is offered to exit setup to create
  76. space/ choose new target.
  77. Arguments:
  78. SifHandle: Handle the txtsetup.sif
  79. TargetRegion: Variable to receive the partition of the Windows NT to install
  80. NULL if not chosen. Caller should not free.
  81. TargetPath: Variable to receive the target path of Windows NT. NULL if
  82. not decided. Caller can free.
  83. SystemPartitionRegion:
  84. Variable to receive the system partition of the Windows NT
  85. NULL if not chosen. Caller should not free.
  86. Return Value:
  87. UpgradeFull: If user chooses to upgrade an NT
  88. UpgradeInstallFresh: If user chooses to install fresh into an existing NT
  89. tree.
  90. DontUpgrade: If user chooses to cancel upgrade and choose a fresh
  91. tree for installation
  92. --*/
  93. {
  94. ENUMUPGRADETYPE UpgradeType;
  95. UPG_PROGRESS_TYPE UpgradeProgressValue;
  96. NTSTATUS NtStatus;
  97. ULONG i,j;
  98. ULONG UpgradeBootSets;
  99. ULONG PidIndex;
  100. PSP_BOOT_ENTRY BootEntry;
  101. PSP_BOOT_ENTRY ChosenBootEntry;
  102. PSP_BOOT_ENTRY MatchedSet = NULL;
  103. ULONG UpgradeOnlyBootSets;
  104. PVOID p;
  105. PWSTR Pid;
  106. ULONG TotalSizeOfFilesOnOsWinnt = 0;
  107. PWSTR UniqueIdFromSif;
  108. PWSTR UniqueIdFromReg;
  109. BOOLEAN Compliant;
  110. BOOLEAN WindowsUpgrade;
  111. BOOLEAN ComplianceChecked;
  112. PWSTR EulaComplete;
  113. DWORD Version = 0, BuildNumber = 0;
  114. DetermineSourceVersionInfo(&Version, &BuildNumber);
  115. //
  116. // If we know we're upgrading NT (chosen during winnt32) then fetch the
  117. // unique id from the parameters file. This will be used later to
  118. // find the system to be upgraded.
  119. //
  120. UniqueIdFromSif = NULL;
  121. p = SpGetSectionKeyIndex(WinntSifHandle,SIF_DATA,WINNT_D_NTUPGRADE_W,0);
  122. if(p && !_wcsicmp(p,WINNT_A_YES_W)) {
  123. UniqueIdFromSif = SpGetSectionKeyIndex(WinntSifHandle,SIF_DATA,SIF_UNIQUEID,0);
  124. if(!UniqueIdFromSif) {
  125. SpFatalSifError(WinntSifHandle,SIF_DATA,SIF_UNIQUEID,0,0);
  126. }
  127. }
  128. //
  129. // Go through all boot sets, looking for upgradeable ones.
  130. //
  131. SpDetermineUniqueAndPresentBootEntries();
  132. UpgradeBootSets = 0;
  133. UpgradeOnlyBootSets = 0;
  134. PidIndex = 0;
  135. for(BootEntry = SpBootEntries; BootEntry != NULL; BootEntry = BootEntry->Next) {
  136. //
  137. // The set's upgradeable flag might already be 0, such as it if was
  138. // a duplicate entry in boot.ini/nv-ram.
  139. // After we've checked for this reset the upgradeable flag for this
  140. // boot set to FALSE in preparation for validating upgreadeability below.
  141. //
  142. if (!BootEntry->Processable) {
  143. continue;
  144. }
  145. BootEntry->Processable = FALSE;
  146. Pid = NULL;
  147. BootEntry->LangId = -1;
  148. //
  149. // Determine various things about the build identified by the
  150. // current boot set (product type -- srv, wks, etc; version and
  151. // build numbers, upgrade progress value, unique id winnt32 put
  152. // in there if any, etc).
  153. //
  154. // Based on the information, we will update the UpgradeableList and
  155. // initialize FailedUpgradeList.
  156. //
  157. NtStatus = SpDetermineProduct(
  158. BootEntry->OsPartitionDiskRegion,
  159. BootEntry->OsDirectory,
  160. &BootEntry->ProductType,
  161. &BootEntry->MajorVersion,
  162. &BootEntry->MinorVersion,
  163. &BootEntry->BuildNumber,
  164. &BootEntry->ProductSuiteMask,
  165. &UpgradeProgressValue,
  166. &UniqueIdFromReg,
  167. &Pid,
  168. NULL, // ignore eval variation flag
  169. &BootEntry->LangId,
  170. &BootEntry->ServicePack
  171. );
  172. if(!NT_SUCCESS(NtStatus)) {
  173. continue;
  174. }
  175. //
  176. // Determine if this installation matches the one we're supposed
  177. // to upgrade (the one the user ran winnt32 on). If this is
  178. // a winnt32 based installation, there is no need to do a
  179. // compliance test as this was already completed during winnt32.
  180. //
  181. if(UniqueIdFromReg) {
  182. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "SETUP: SpFindNtToUpgrade: BootEntry = %p, RegId = %S, UniqueId = %S\n", BootEntry, UniqueIdFromReg, UniqueIdFromSif));
  183. if(UniqueIdFromSif && (MatchedSet == NULL)
  184. && !wcscmp(UniqueIdFromSif,UniqueIdFromReg)) {
  185. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "SETUP: SpFindNtToUpgrade: found a match!\n"));
  186. MatchedSet = BootEntry;
  187. BootEntry->Processable = TRUE;
  188. }
  189. SpMemFree(UniqueIdFromReg);
  190. UniqueIdFromReg = NULL;
  191. }
  192. if (BootEntry->Processable == FALSE) {
  193. //
  194. // this is set to TRUE if this is the build we ran winnt32 upgrade
  195. // from -- in all other cases we need to do a compliance test
  196. // to determine if this is a valid build to upgrade
  197. //
  198. Compliant = pSpIsCompliant( Version,
  199. BuildNumber,
  200. BootEntry->OsPartitionDiskRegion,
  201. BootEntry->OsDirectory,
  202. &BootEntry->UpgradeOnlyCompliance );
  203. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "SETUP: SpFindNtToUpgrade says UpgradeableList[%p] compliance test is %s, upgrade only : %s\n",
  204. BootEntry,
  205. Compliant ? "TRUE" : "FALSE" ,
  206. BootEntry->UpgradeOnlyCompliance ? "TRUE" : "FALSE"
  207. ));
  208. BootEntry->Processable = Compliant;
  209. if (BootEntry->UpgradeOnlyCompliance) {
  210. UpgradeOnlyBootSets++;
  211. }
  212. }
  213. if(BootEntry->Processable) {
  214. UpgradeBootSets++;
  215. //
  216. // Save the Pid only if it is Pid20
  217. //
  218. if(wcslen(Pid) == 20) {
  219. BootEntry->Pid20Array = Pid;
  220. } else {
  221. SpMemFree(Pid);
  222. }
  223. } else {
  224. SpMemFree(Pid);
  225. }
  226. BootEntry->FailedUpgrade = (UpgradeProgressValue == UpgradeInProgress);
  227. }
  228. //
  229. // Winnt32 displays the EULA which signifies that compliance checking has been
  230. // completed
  231. //
  232. EulaComplete = SpGetSectionKeyIndex(WinntSifHandle, SIF_DATA,WINNT_D_EULADONE_W, 0);
  233. if(EulaComplete && SpStringToLong(EulaComplete, NULL, 10)) {
  234. ComplianceChecked = TRUE;
  235. } else {
  236. ComplianceChecked = FALSE;
  237. }
  238. //
  239. // don't try to validate if we are upgrading a Win3.X or Win9X installation
  240. //
  241. #ifdef _X86_
  242. WindowsUpgrade = SpIsWindowsUpgrade(WinntSifHandle);
  243. #else
  244. WindowsUpgrade = FALSE;
  245. #endif // _X86_
  246. //
  247. // In step-up mode, we need to ensure that the user has a qualifiying product.
  248. //
  249. //
  250. // If we couldn't find it on the machine, go perform additional validation
  251. // steps.
  252. //
  253. if(StepUpMode && !UpgradeBootSets && !WindowsUpgrade && !ComplianceChecked) {
  254. pSpStepUpValidate();
  255. }
  256. //
  257. // If we are supposed to be upgrading NT then make sure we found
  258. // the system we're supposed to upgrade.
  259. //
  260. if(UniqueIdFromSif) {
  261. if(MatchedSet == NULL) {
  262. SpCantFindBuildToUpgrade();
  263. }
  264. ChosenBootEntry = MatchedSet;
  265. UpgradeType = UpgradeFull;
  266. OldMajorVersion = ChosenBootEntry->MajorVersion;
  267. OldMinorVersion = ChosenBootEntry->MinorVersion;
  268. #ifndef _X86_
  269. //
  270. // On non-x86 platforms, especially alpha machines that in general
  271. // have small system partitions (~3 MB), we compute the size
  272. // of the files on \os\winnt (ie osloader.exe and hal.dll),
  273. // and consider this size as available disk space. We can do this
  274. // since these files will be overwritten by the new ones.
  275. // This fixes the problem that we see on Alpha, when the system
  276. // partition is too full.
  277. //
  278. SpFindSizeOfFilesInOsWinnt(
  279. SifHandle,
  280. ChosenBootEntry->LoaderPartitionDiskRegion,
  281. &TotalSizeOfFilesOnOsWinnt
  282. );
  283. //
  284. // Transform the size into KB
  285. //
  286. TotalSizeOfFilesOnOsWinnt /= 1024;
  287. #endif
  288. //
  289. // If a previous upgrade attempt on this build failed
  290. // (say the power went out in the middle) then we will try to
  291. // upgrade it again. We can't offer to install a fresh build
  292. // because we're not sure we can reliably complete it
  293. // (for example winnt32.exe might copy down only a subset of files
  294. // across the net when it knows the user ios upgrading and so
  295. // initial install could fail because of missing files, etc).
  296. //
  297. // If the disk is too full then the user is hosed. Tell him and exit.
  298. //
  299. if(ChosenBootEntry->FailedUpgrade) {
  300. SppResumingFailedUpgrade(
  301. ChosenBootEntry->OsPartitionDiskRegion,
  302. ChosenBootEntry->OsDirectory,
  303. ChosenBootEntry->FriendlyName,
  304. FALSE
  305. );
  306. }
  307. } else {
  308. //
  309. // Not upgrading. However for PSS we allow the user to upgrade a build
  310. // "in place" as a sort of emergency repair thing. The build has to be
  311. // the same build number as the one we're installing.
  312. //
  313. UpgradeType = DontUpgrade;
  314. //
  315. // Also, if the user is upgrading Windows 95 or we're in unattended mode
  316. // then we don't ask the user anything.
  317. //
  318. p = SpGetSectionKeyIndex(WinntSifHandle,SIF_DATA,WINNT_D_WIN95UPGRADE_W,0);
  319. if(!UnattendedOperation && (!p || _wcsicmp(p,WINNT_A_YES_W))) {
  320. //
  321. // Eliminate from the upgradeable list those builds which
  322. // don't match.
  323. //
  324. j = 0;
  325. for(BootEntry = SpBootEntries; BootEntry != NULL; BootEntry = BootEntry->Next) {
  326. if(BootEntry->Processable) {
  327. if (!SpDoBuildsMatch(
  328. SifHandle,
  329. BootEntry->BuildNumber,
  330. BootEntry->ProductType,
  331. BootEntry->ProductSuiteMask,
  332. AdvancedServer,
  333. SuiteType,
  334. BootEntry->LangId)) {
  335. BootEntry->Processable = FALSE;
  336. j++;
  337. }
  338. }
  339. }
  340. UpgradeBootSets -= j;
  341. if(UpgradeBootSets) {
  342. //
  343. // Find out if the user wants to "upgrade" one of these.
  344. //
  345. UpgradeType = SppSelectNTToRepairByUpgrade(
  346. &ChosenBootEntry
  347. );
  348. #ifndef _X86_
  349. SpFindSizeOfFilesInOsWinnt(
  350. SifHandle,
  351. ChosenBootEntry->LoaderPartitionDiskRegion,
  352. &TotalSizeOfFilesOnOsWinnt
  353. );
  354. TotalSizeOfFilesOnOsWinnt /= 1024;
  355. #endif
  356. if(UpgradeType == UpgradeFull) {
  357. //
  358. // Check for resume case and inform user.
  359. //
  360. if(ChosenBootEntry->FailedUpgrade) {
  361. //
  362. // If user cancelled then lets try to do a
  363. // clean install
  364. //
  365. if (!SppResumingFailedUpgrade(
  366. ChosenBootEntry->OsPartitionDiskRegion,
  367. ChosenBootEntry->OsDirectory,
  368. ChosenBootEntry->FriendlyName,
  369. TRUE
  370. )) {
  371. UpgradeType = DontUpgrade;
  372. }
  373. } else {
  374. //
  375. // Everything is OK.
  376. //
  377. OldMajorVersion = ChosenBootEntry->MajorVersion;
  378. OldMinorVersion = ChosenBootEntry->MinorVersion;
  379. }
  380. }
  381. }
  382. }
  383. }
  384. //
  385. // Depending on upgrade selection made do the setup needed before
  386. // we do the upgrade
  387. //
  388. if(UpgradeType == UpgradeFull) {
  389. PWSTR p1,p2,p3;
  390. //
  391. // Set the upgrade status to upgrading in the current system hive
  392. //
  393. SpSetUpgradeStatus(
  394. ChosenBootEntry->OsPartitionDiskRegion,
  395. ChosenBootEntry->OsDirectory,
  396. UpgradeInProgress
  397. );
  398. //
  399. // Return the region we are installing onto
  400. //
  401. *TargetRegion = ChosenBootEntry->OsPartitionDiskRegion;
  402. *TargetPath = SpDupStringW(ChosenBootEntry->OsDirectory);
  403. *SystemPartitionRegion = ChosenBootEntry->LoaderPartitionDiskRegion;
  404. StandardServerUpgrade = ( AdvancedServer &&
  405. ( ChosenBootEntry->ProductType == NtProductWinNt ) ||
  406. ( ChosenBootEntry->ProductType == NtProductServer )
  407. );
  408. //
  409. // Process the osloader variable to extract the system partition path.
  410. // The var could be of the form ...partition(1)\os\nt\... or
  411. // ...partition(1)os\nt\...
  412. // So we search forward for the first \ and then backwards for
  413. // the closest ) to find the start of the directory. We then
  414. // search backwards in the resulting string for the last \ to find
  415. // the end of the directory.
  416. //
  417. p1 = ChosenBootEntry->LoaderFile;
  418. p2 = wcsrchr(p1, L'\\');
  419. if (p2 == NULL) {
  420. p2 = p1;
  421. }
  422. i = (ULONG)(p2 - p1);
  423. if(i == 0) {
  424. *SystemPartitionDirectory = SpDupStringW(L"");
  425. } else {
  426. p2 = p3 = SpMemAlloc((i+2)*sizeof(WCHAR));
  427. ASSERT(p3);
  428. if(*p1 != L'\\') {
  429. *p3++ = L'\\';
  430. }
  431. wcsncpy(p3, p1, i);
  432. p3[i] = 0;
  433. *SystemPartitionDirectory = p2;
  434. }
  435. }
  436. //
  437. // Clean up and return,
  438. //
  439. CLEAR_CLIENT_SCREEN();
  440. return(UpgradeType);
  441. }
  442. BOOLEAN
  443. pSpGetCdInstallType(
  444. IN PWSTR PathToCd,
  445. OUT PULONG CdInstallType,
  446. OUT PULONG CdInstallVersion
  447. )
  448. {
  449. #define BuildMatch(_filename_,_type_,_ver_) \
  450. FileName = _filename_; \
  451. b = SpNFilesExist(PathToCd,&FileName,1,FALSE); \
  452. if (b) { \
  453. *CdInstallType = _type_; \
  454. *CdInstallVersion = _ver_; \
  455. } \
  456. //return(TRUE); \
  457. BOOLEAN b;
  458. //
  459. // Directories that are present on all known NT CD-ROM's.
  460. //
  461. //
  462. // ISSUE:2000/27/07:vijayj: this code seems really busted. In looking at a handful of cd's, none
  463. // of the nt cd's seem to conform to these rules listed
  464. // also looks like there might be "per architecture" tag files as well.
  465. //
  466. //Check for both dirs to exist? Definitely nt4 has these. Some pre RTM w2k cds have them
  467. PWSTR ListAllPreNT5[] = { L"alpha", L"i386" };
  468. //wk2 RTM , whistler and nt4 have thse two dirs. They differ by tag file.
  469. PWSTR ListAllNT5[] = { L"i386", L"support" };
  470. PWSTR ListAllNec98[] = { L"pc98",L"support" }; //NEC98
  471. PWSTR ListAllEntNT4[] = { L"alpha", L"i386", L"SP3" };
  472. //
  473. // Directories which must be present if a CD is a 3.51 or a 4.0 CD-ROM,
  474. // workstation or server. Note that the ppc directory distinguishes
  475. // 3.51 from 3.5.
  476. //
  477. PWSTR List351_40[] = { L"mips", L"ppc" };
  478. //
  479. // directories which must be present if a CD is a win95 or win98 cd-rom.
  480. //
  481. PWSTR ListWin95[] = { L"win95", L"drivers" };
  482. PWSTR ListWin98[] = { L"win98", L"drivers" };
  483. PWSTR ListWinME[] = { L"win9x", L"drivers" };
  484. PWSTR FileName;
  485. //
  486. // check for NT4 enterprise
  487. //
  488. b = SpNFilesExist(
  489. PathToCd,
  490. ListAllEntNT4,
  491. sizeof(ListAllEntNT4)/sizeof(ListAllEntNT4[0]),
  492. TRUE
  493. );
  494. if (b) {
  495. BuildMatch(L"cdrom_s.40", COMPLIANCE_INSTALLTYPE_NTSE, 400);
  496. if (b) {
  497. return(TRUE);
  498. }
  499. }
  500. //
  501. // check for various subsets of NT < NT5
  502. //
  503. b = SpNFilesExist(
  504. PathToCd,
  505. (!IsNEC_98) ? ListAllPreNT5 : ListAllNec98, //NEC98
  506. (!IsNEC_98) ? sizeof(ListAllPreNT5)/sizeof(ListAllPreNT5[0]) : sizeof(ListAllNec98)/sizeof(ListAllNec98[0]), //NEC98
  507. TRUE
  508. );
  509. if(b) {
  510. //
  511. // hydra (terminal server) is a special case (since it does not
  512. // have mips and ppc directory).
  513. //
  514. BuildMatch(L"cdrom_ts.40", COMPLIANCE_INSTALLTYPE_NTS, 400);
  515. if (b) {
  516. return TRUE;
  517. }
  518. //
  519. // OK, it could be an NT CD of some kind, but it could be
  520. // 3.1, 3.5, 3.51, 4.0. It could also be
  521. // server or workstation. Narrow down to 3.51/4.0.
  522. //
  523. b = SpNFilesExist(PathToCd,List351_40,
  524. sizeof(List351_40)/sizeof(List351_40[0]),TRUE);
  525. if(b) {
  526. //
  527. // If we get here, we know it can only be either 3.51 or 4.0.
  528. // Look for 3.51.
  529. //
  530. BuildMatch(L"cdrom.s", COMPLIANCE_INSTALLTYPE_NTS, 351);
  531. if (b) {
  532. return(TRUE);
  533. }
  534. BuildMatch(L"cdrom.w", COMPLIANCE_INSTALLTYPE_NTW, 351);
  535. if (b) {
  536. return(TRUE);
  537. }
  538. //
  539. // Look for 4.0.
  540. //
  541. BuildMatch(L"cdrom_s.40", COMPLIANCE_INSTALLTYPE_NTS, 400);
  542. if (b) {
  543. return(TRUE);
  544. }
  545. BuildMatch(L"cdrom_w.40", COMPLIANCE_INSTALLTYPE_NTW, 400);
  546. if (b) {
  547. return(TRUE);
  548. }
  549. } else {
  550. //
  551. // Find out if its one of the NT 4.0 service pack CDs
  552. //
  553. BuildMatch(L"cdrom_s.40", COMPLIANCE_INSTALLTYPE_NTS, 400);
  554. if (b) {
  555. return(TRUE);
  556. }
  557. BuildMatch(L"cdrom_w.40", COMPLIANCE_INSTALLTYPE_NTW, 400);
  558. if (b) {
  559. return(TRUE);
  560. }
  561. //
  562. // Not 3.51 or 4.0. Check for 5.0 beta 1 and beta2
  563. //
  564. //How is this possible to be 5.0 unless alpha dir exists on cd.
  565. //
  566. // Post beta 1 the tag files changed to support per architecture tags
  567. // but beta 2 still has alpha directories
  568. //
  569. // We could possibly just check to see if cdrom_w.50 isn't
  570. // there, but the NT3.1 CD would then pass so we need to
  571. // check explicitly for the 5.0 beta CDs.
  572. //
  573. BuildMatch(L"cdrom_s.5b1", COMPLIANCE_INSTALLTYPE_NTS, 500);
  574. if (b) {
  575. return(FALSE); //eval
  576. }
  577. BuildMatch(L"cdrom_w.5b1", COMPLIANCE_INSTALLTYPE_NTW, 500);
  578. if (b) {
  579. return(FALSE); //eval
  580. }
  581. BuildMatch(L"cdrom_s.5b2", COMPLIANCE_INSTALLTYPE_NTS, 500);
  582. if (b) {
  583. return(FALSE); //eval
  584. }
  585. BuildMatch(L"cdrom_w.5b2", COMPLIANCE_INSTALLTYPE_NTW, 500);
  586. if (b) {
  587. return(FALSE); //eval
  588. }
  589. BuildMatch(L"cdrom_s.5b3", COMPLIANCE_INSTALLTYPE_NTS, 500);
  590. if (b) {
  591. return(FALSE); //eval
  592. }
  593. BuildMatch(L"cdrom_w.5b3", COMPLIANCE_INSTALLTYPE_NTW, 500);
  594. if (b) {
  595. return(FALSE); //eval
  596. }
  597. BuildMatch(L"cdrom_is.5b2", COMPLIANCE_INSTALLTYPE_NTS, 500);
  598. if (b) {
  599. return(FALSE); //eval
  600. }
  601. BuildMatch(L"cdrom_iw.5b2", COMPLIANCE_INSTALLTYPE_NTW, 500);
  602. if (b) {
  603. return(FALSE); //eval
  604. }
  605. BuildMatch(L"cdrom_ie.5b2", COMPLIANCE_INSTALLTYPE_NTSE, 500);
  606. if (b) {
  607. return(FALSE); //eval
  608. }
  609. //Do we need to check for eval nt5.1 here? No since alpha dir doesnt exist on the cds.
  610. //
  611. // if we made it this far, it must be nt 3.1/3.5
  612. //
  613. // we just mark the version as 3.5 since we don't allow upgradescd o
  614. // from either type of install.
  615. //
  616. *CdInstallType = COMPLIANCE_INSTALLTYPE_NTW;
  617. *CdInstallVersion = 350;
  618. return(TRUE);
  619. }
  620. }
  621. //
  622. // look for various nt5 beta cds.
  623. // // Could also be nt5.1 since cd also contains same dir.
  624. //
  625. // note that we don't check 5.0 retail since that would allow the retail CD to
  626. // validate itself, which defeats the purpose entirely.
  627. //
  628. // Post NT5 beta 1 the tag files changed to support per architecture tags so we have
  629. // a massive ifdef below
  630. //
  631. //
  632. b = SpNFilesExist(
  633. PathToCd,
  634. ListAllNT5,
  635. sizeof(ListAllNT5)/sizeof(ListAllNT5[0]),
  636. TRUE
  637. );
  638. if (b) {
  639. //
  640. // we might have some flavour of NT5 beta cd, but we're not sure which one
  641. //
  642. BuildMatch(L"cdrom_s.5b1", COMPLIANCE_INSTALLTYPE_NTS, 500);
  643. if (b) {
  644. return(FALSE); //eval
  645. }
  646. BuildMatch(L"cdrom_w.5b1", COMPLIANCE_INSTALLTYPE_NTW, 500);
  647. if (b) {
  648. return(FALSE); //eval
  649. }
  650. BuildMatch(L"cdrom_is.5b2", COMPLIANCE_INSTALLTYPE_NTS, 500);
  651. if (b) {
  652. return(FALSE); //eval
  653. }
  654. BuildMatch(L"cdrom_iw.5b2", COMPLIANCE_INSTALLTYPE_NTW, 500);
  655. if (b) {
  656. return(FALSE); //eval
  657. }
  658. BuildMatch(L"cdrom_ie.5b2", COMPLIANCE_INSTALLTYPE_NTSE, 500);
  659. if (b) {
  660. return(FALSE); //eval
  661. }
  662. BuildMatch(L"cdrom_is.5b3", COMPLIANCE_INSTALLTYPE_NTS, 500);
  663. if (b) {
  664. return(FALSE); //eval
  665. }
  666. BuildMatch(L"cdrom_iw.5b3", COMPLIANCE_INSTALLTYPE_NTW, 500);
  667. if (b) {
  668. return(FALSE); //eval
  669. }
  670. BuildMatch(L"cdrom_ie.5b3", COMPLIANCE_INSTALLTYPE_NTSE, 500);
  671. if (b) {
  672. return(FALSE); //eval
  673. }
  674. BuildMatch(L"cdrom_nt.51", COMPLIANCE_INSTALLTYPE_NTW, 501);
  675. if (b) {
  676. return(FALSE); //pre beta1 whistler
  677. }
  678. BuildMatch(L"wen51.b1", COMPLIANCE_INSTALLTYPE_NTW, 501);
  679. if (b) {
  680. return(FALSE); //beta 1 whistler
  681. }
  682. BuildMatch(L"win51.b2", COMPLIANCE_INSTALLTYPE_NTW, 501);
  683. if (b) {
  684. return(FALSE); //beta 1 whistler
  685. }
  686. BuildMatch(L"win51.rc1", COMPLIANCE_INSTALLTYPE_NTW, 501);
  687. if (b) {
  688. return(FALSE); //rc1 whistler
  689. }
  690. }
  691. //
  692. // check for win95
  693. //
  694. b = SpNFilesExist(PathToCd, ListWin95, sizeof(ListWin95)/sizeof(ListWin95[0]),TRUE );
  695. if (b) {
  696. *CdInstallType = COMPLIANCE_INSTALLTYPE_WIN9X;
  697. *CdInstallVersion = 950;
  698. return TRUE;
  699. }
  700. //
  701. // check for win98
  702. //
  703. b = SpNFilesExist(PathToCd, ListWin98, sizeof(ListWin98)/sizeof(ListWin98[0]),TRUE );
  704. if (b) {
  705. *CdInstallType = COMPLIANCE_INSTALLTYPE_WIN9X;
  706. *CdInstallVersion = 1998;
  707. return TRUE;
  708. }
  709. //
  710. // check for winME
  711. //
  712. b = SpNFilesExist(PathToCd, ListWinME, sizeof(ListWinME)/sizeof(ListWinME[0]),TRUE );
  713. if (b) {
  714. *CdInstallType = COMPLIANCE_INSTALLTYPE_WIN9X;
  715. *CdInstallVersion = 3000;
  716. return TRUE;
  717. }
  718. //At this point we have rejected w2k beta cds. However we need to reject wk2 eval cds.
  719. //We should accept w2k stepup media so the next check is only for 5.1
  720. //Need to accept nt5.1 eval cds.!!Ask rajj to verify.
  721. // Need to reject nt5.1 step-up cds.
  722. // Need to reject nt5.1 rtm cds? Looks like we accept RTM FPP?
  723. //
  724. // could be NT 5.1 CD-ROM (make sure its not eval media)
  725. //
  726. if (!b) { //check is not needed.
  727. NTSTATUS Status;
  728. CCMEDIA MediaObj;
  729. WCHAR InfDir[MAX_PATH];
  730. wcscpy(InfDir, PathToCd);
  731. SpConcatenatePaths(InfDir, (IsNEC_98 ? ListAllNec98[0] : ListAllNT5[0]));
  732. Status = SpGetMediaDetails(InfDir, &MediaObj);
  733. if (NT_SUCCESS(Status)) {
  734. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "Type=%lx, Variation=%lx, Version=%lx, Build=%lx, SetupMedia=%s\n",
  735. MediaObj.SourceType, MediaObj.SourceVariation, MediaObj.Version,
  736. MediaObj.BuildNumber,
  737. (MediaObj.StepUpMedia ? L"True" : L"False")));
  738. if( (MediaObj.Version == 500) &&
  739. (MediaObj.SourceVariation != COMPLIANCE_INSTALLVAR_EVAL) &&
  740. (MediaObj.BuildNumber == 2195)) {
  741. *CdInstallType = MediaObj.SourceType;
  742. *CdInstallVersion = MediaObj.Version;
  743. return TRUE;
  744. }
  745. // At this point we should be current media 5.1
  746. if( MediaObj.Version == 501 ) {
  747. if( MediaObj.SourceVariation != COMPLIANCE_INSTALLVAR_EVAL) {
  748. if( MediaObj.StepUpMedia == FALSE) {
  749. *CdInstallType = MediaObj.SourceType;
  750. *CdInstallVersion = MediaObj.Version;
  751. return TRUE;
  752. } else {
  753. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "pSpGetCdInstallType: SpGetMediaDetails succeeded but STEPUP media"
  754. " cannot be used for validation\n", Status));
  755. }
  756. } else {
  757. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "pSpGetCdInstallType: SpGetMediaDetails succeeded but Eval media\n",
  758. Status));
  759. }
  760. } else {
  761. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "pSpGetCdInstallType: SpGetMediaDetails succeeded but unrecognized version\n",
  762. Status));
  763. }
  764. } else {
  765. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "pSpGetCdInstallType: SpGetMediaDetails failed with %lx error code\n",
  766. Status));
  767. }
  768. }
  769. //
  770. // not any system CD that we know about
  771. //
  772. return(FALSE);
  773. }
  774. VOID
  775. pSpStepUpValidate(
  776. VOID
  777. )
  778. {
  779. ULONG CdCount;
  780. ULONG i;
  781. BOOLEAN b;
  782. ULONG Prompt,SecondaryPrompt;
  783. ULONG ValidKeys[3] = { ASCI_CR, KEY_F3, 0 };
  784. LARGE_INTEGER DelayTime;
  785. PWSTR FileName;
  786. ULONG SourceSkuId;
  787. ULONG DontCare;
  788. ULONG CdInstallType;
  789. ULONG CdInstallVersion;
  790. //
  791. // Directories that are present on all known NT CD-ROM's.
  792. //
  793. PWSTR ListAll[] = { L"alpha", L"i386" };
  794. PWSTR ListAllNec98[] = { L"pc98",L"support" }; //NEC98
  795. //
  796. // Directories which must be present if a CD is a 3.51 or a 4.0 CD-ROM,
  797. // workstation or server. Note that the ppc directory distinguishes
  798. // 3.51 from 3.5.
  799. //
  800. PWSTR List351_40[] = { L"mips", L"ppc" };
  801. PWSTR ListWin95[] = { L"win95", L"autorun" };
  802. PWSTR ListWin98[] = { L"win98", L"autorun" };
  803. SourceSkuId = DetermineSourceProduct(&DontCare,NULL);
  804. Prompt = SP_SCRN_STEP_UP_NO_QUALIFY;
  805. switch (SourceSkuId) {
  806. case COMPLIANCE_SKU_NTW32U:
  807. SecondaryPrompt = SP_SCRN_STEP_UP_PROMPT_WKS;
  808. break;
  809. case COMPLIANCE_SKU_NTSU:
  810. SecondaryPrompt = SP_SCRN_STEP_UP_PROMPT_SRV;
  811. break;
  812. case COMPLIANCE_SKU_NTSEU:
  813. SecondaryPrompt = SP_SCRN_STEP_UP_PROMPT_ENT;
  814. break;
  815. default:
  816. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "setup: Unexpected SKU %x, defaulting to workstation\n", SourceSkuId));
  817. SecondaryPrompt = SP_SCRN_STEP_UP_PROMPT_WKS;
  818. break;
  819. }
  820. #if 0
  821. //
  822. // ntw upgrade is a special case because you have either win95 or an old NTW to
  823. // upgrade from
  824. //
  825. // might have to prompt for floppies
  826. //
  827. if (SourceSkuId == COMPLIANCE_SKU_NTWU) {
  828. }
  829. //
  830. // See if there is a CD-ROM drive. If not we can't continue.
  831. //
  832. else
  833. #endif
  834. if(CdCount = IoGetConfigurationInformation()->CdRomCount) {
  835. do {
  836. //
  837. // Tell the user what's going on. This screen also contains
  838. // a prompt to insert a qualifying CD-ROM.
  839. //
  840. while(1) {
  841. SpStartScreen(Prompt,3,HEADER_HEIGHT+1,FALSE,FALSE,DEFAULT_ATTRIBUTE);
  842. SpContinueScreen(
  843. SecondaryPrompt,
  844. 3,
  845. 1,
  846. FALSE,
  847. DEFAULT_ATTRIBUTE
  848. );
  849. SpContinueScreen(SP_SCRN_STEP_UP_INSTRUCTIONS,3,1,FALSE,DEFAULT_ATTRIBUTE);
  850. SpDisplayStatusOptions(
  851. DEFAULT_STATUS_ATTRIBUTE,
  852. SP_STAT_ENTER_EQUALS_CONTINUE,
  853. SP_STAT_F3_EQUALS_EXIT,
  854. 0
  855. );
  856. if(SpWaitValidKey(ValidKeys,NULL,NULL) == KEY_F3) {
  857. SpConfirmExit();
  858. } else {
  859. break;
  860. }
  861. }
  862. CLEAR_CLIENT_SCREEN();
  863. SpDisplayStatusText(SP_STAT_PLEASE_WAIT,DEFAULT_STATUS_ATTRIBUTE);
  864. //
  865. // Wait 5 sec for the CD to become ready
  866. //
  867. DelayTime.HighPart = -1;
  868. DelayTime.LowPart = (ULONG)(-50000000);
  869. KeDelayExecutionThread(KernelMode,FALSE,&DelayTime);
  870. //
  871. // Check for relevent files/dirs on each CD-ROM drive.
  872. //
  873. for(i=0; i<CdCount; i++) {
  874. swprintf(TemporaryBuffer,L"\\Device\\Cdrom%u",i);
  875. if (pSpGetCdInstallType(TemporaryBuffer, &CdInstallType, &CdInstallVersion) ) {
  876. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "cd type : %x cd version : %d\n", CdInstallType, CdInstallVersion ));
  877. switch (SourceSkuId) {
  878. case COMPLIANCE_SKU_NTWPU:
  879. case COMPLIANCE_SKU_NTW32U:
  880. if ( (CdInstallType == COMPLIANCE_INSTALLTYPE_WIN9X) ||
  881. ( ((CdInstallType == COMPLIANCE_INSTALLTYPE_NTW) ||
  882. (CdInstallType == COMPLIANCE_INSTALLTYPE_NTWP)) &&
  883. (CdInstallVersion > 350)) ) {
  884. return;
  885. }
  886. break;
  887. case COMPLIANCE_SKU_NTSU:
  888. if ( (CdInstallType == COMPLIANCE_INSTALLTYPE_NTS) &&
  889. (CdInstallVersion > 350) ) {
  890. return;
  891. }
  892. break;
  893. case COMPLIANCE_SKU_NTSEU:
  894. if (CdInstallType == COMPLIANCE_INSTALLTYPE_NTSE) {
  895. return;
  896. }
  897. break;
  898. default:
  899. return;
  900. }
  901. }
  902. }
  903. //
  904. // If we get here then the CD the user inserted is bogus
  905. // or the user didn't insert a CD. Reprompt. The while loop
  906. // condition makes this essentially an infinite loop.
  907. //
  908. } while(Prompt = SP_SCRN_STEP_UP_BAD_CD);
  909. }
  910. SpStartScreen(
  911. SP_SCRN_STEP_UP_FATAL,
  912. 3,
  913. HEADER_HEIGHT+1,
  914. FALSE,
  915. FALSE,
  916. DEFAULT_ATTRIBUTE
  917. );
  918. SpDisplayStatusOptions(DEFAULT_STATUS_ATTRIBUTE,SP_STAT_F3_EQUALS_EXIT,0);
  919. SpInputDrain();
  920. while(SpInputGetKeypress() != KEY_F3);
  921. SpDone(0,FALSE,FALSE);
  922. }
  923. BOOLEAN
  924. SppResumingFailedUpgrade(
  925. IN PDISK_REGION Region,
  926. IN LPCWSTR OsLoadFileName,
  927. IN LPCWSTR LoadIdentifier,
  928. IN BOOLEAN AllowCancel
  929. )
  930. /*++
  931. Routine Description:
  932. Simple routine to inform the user that setup noticed that the build
  933. chosen for upgrade had been upgraded before, but the upgrade attempt
  934. failed. The user can continue or exit. If he continues the build will be
  935. upgraded (again).
  936. Arguments:
  937. Region - supplies region containing the build being upgraded
  938. OsLoadFileName - supplies ARC OSLOADFILENAME var for the build (ie, sysroot)
  939. LoadIdentifier - supplies ARC LOADIDENTIFIER for the build (ie, human-
  940. readable description).
  941. AllowCancel - Indicates whether user can cancel out of this or not
  942. Return Value:
  943. TRUE, if the user wants to continue and attempt to upgrade again else
  944. FALSE.
  945. --*/
  946. {
  947. ULONG ValidKeys[] = { KEY_F3, ASCI_CR, 0, 0 };
  948. ULONG c;
  949. DRIVELTR_STRING UpgDriveLetter;
  950. ULONG MsgId;
  951. ULONG EscStatusId;
  952. BOOLEAN AllowUpgrade = TRUE;
  953. ASSERT(Region->PartitionedSpace);
  954. ASSERT(wcslen(OsLoadFileName) >= 2);
  955. SpGetUpgDriveLetter(Region->DriveLetter,
  956. UpgDriveLetter,
  957. sizeof(UpgDriveLetter),
  958. FALSE);
  959. if (AllowCancel) {
  960. ValidKeys[2] = ASCI_ESC;
  961. MsgId = SP_SCRN_WINNT_FAILED_UPGRADE_ESC;
  962. EscStatusId = SP_STAT_ESC_EQUALS_CLEAN_INSTALL;
  963. } else {
  964. MsgId = SP_SCRN_WINNT_FAILED_UPGRADE;
  965. EscStatusId = 0;
  966. }
  967. do {
  968. SpStartScreen(
  969. MsgId,
  970. 3,
  971. HEADER_HEIGHT+1,
  972. FALSE,
  973. FALSE,
  974. DEFAULT_ATTRIBUTE,
  975. UpgDriveLetter,
  976. OsLoadFileName,
  977. LoadIdentifier
  978. );
  979. SpDisplayStatusOptions(
  980. DEFAULT_STATUS_ATTRIBUTE,
  981. SP_STAT_F3_EQUALS_EXIT,
  982. SP_STAT_ENTER_EQUALS_UPGRADE,
  983. EscStatusId,
  984. 0
  985. );
  986. c = SpWaitValidKey(ValidKeys, NULL, NULL);
  987. switch (c) {
  988. case KEY_F3:
  989. SpConfirmExit();
  990. break;
  991. case ASCI_ESC:
  992. AllowUpgrade = FALSE;
  993. break;
  994. default:
  995. break;
  996. }
  997. } while (c == KEY_F3);
  998. return AllowUpgrade;
  999. }
  1000. #define MAX_INT_STRING 30
  1001. VOID
  1002. SppUpgradeDiskFull(
  1003. IN PDISK_REGION OsRegion,
  1004. IN LPCWSTR OsLoadFileName,
  1005. IN LPCWSTR LoadIdentifier,
  1006. IN PDISK_REGION SysPartRegion,
  1007. IN ULONG MinOsFree,
  1008. IN ULONG MinSysFree,
  1009. IN BOOLEAN Fatal
  1010. )
  1011. /*++
  1012. Routine Description:
  1013. Inform the user that the nt tree chosen for upgrade can't be upgraded
  1014. because the partition is too full.
  1015. Arguments:
  1016. OsRegion - supplies region containing the nt tree.
  1017. OsLoadFileName - supplies ARC OSLOADFILENAME var for the build (ie, sysroot)
  1018. LoadIdentifier - supplies ARC LOADIDENTIFIER for the build (ie, human-
  1019. readable description).
  1020. SysPartRegion - supplies the region that is the ARC system partition
  1021. for the build being upgraded
  1022. MinOsFree - supplies the size in KB of the minimum amount of free space
  1023. we require before attempting upgrade
  1024. MinSysFree - supplies the size in KB of the minimum amount of free space
  1025. we require on the system partition before attempting upgrade.
  1026. Fatal - if TRUE then the only option is exit. If FALSE then this routine
  1027. can return to its caller.
  1028. Return Value:
  1029. None. MAY NOT RETURN, depending on Fatal.
  1030. --*/
  1031. {
  1032. ULONG ValidKeys[] = { KEY_F3,0,0 };
  1033. PWCHAR Drive1, Drive2;
  1034. DRIVELTR_STRING OsRgnDrive, OsRgnDriveFull, SysRgnDriveFull;
  1035. WCHAR Drive1Free[MAX_INT_STRING], Drive1FreeNeeded[MAX_INT_STRING];
  1036. WCHAR Drive2Free[MAX_INT_STRING], Drive2FreeNeeded[MAX_INT_STRING];
  1037. BOOLEAN FirstDefined = FALSE, SecondDefined = FALSE;
  1038. ASSERT(OsRegion->PartitionedSpace);
  1039. ASSERT(SysPartRegion->PartitionedSpace);
  1040. ASSERT(wcslen(OsLoadFileName) >= 2);
  1041. SpGetUpgDriveLetter(OsRegion->DriveLetter,OsRgnDrive,sizeof(OsRgnDrive),FALSE);
  1042. if((OsRegion == SysPartRegion) || (OsRegion->FreeSpaceKB < MinOsFree)) {
  1043. //
  1044. // Then we'll be needing the full (colon added) version of
  1045. // the drive letter
  1046. //
  1047. SpGetUpgDriveLetter(OsRegion->DriveLetter,OsRgnDriveFull,sizeof(OsRgnDrive),TRUE);
  1048. }
  1049. if(OsRegion == SysPartRegion) {
  1050. Drive1 = OsRgnDriveFull;
  1051. swprintf(Drive1Free,L"%d",OsRegion->FreeSpaceKB);
  1052. swprintf(Drive1FreeNeeded,L"%d",MinOsFree);
  1053. FirstDefined = TRUE;
  1054. } else {
  1055. if(SysPartRegion->FreeSpaceKB < MinSysFree) {
  1056. SpGetUpgDriveLetter(SysPartRegion->DriveLetter,SysRgnDriveFull,sizeof(SysRgnDriveFull),TRUE);
  1057. Drive1 = SysRgnDriveFull;
  1058. swprintf(Drive1Free,L"%d",SysPartRegion->FreeSpaceKB);
  1059. swprintf(Drive1FreeNeeded,L"%d",MinSysFree);
  1060. FirstDefined = TRUE;
  1061. }
  1062. if(OsRegion->FreeSpaceKB < MinOsFree) {
  1063. if(!FirstDefined) {
  1064. Drive1 = OsRgnDriveFull;
  1065. swprintf(Drive1Free,L"%d",OsRegion->FreeSpaceKB);
  1066. swprintf(Drive1FreeNeeded,L"%d",MinOsFree);
  1067. FirstDefined = TRUE;
  1068. } else {
  1069. Drive2 = OsRgnDriveFull;
  1070. swprintf(Drive2Free,L"%d",OsRegion->FreeSpaceKB);
  1071. swprintf(Drive2FreeNeeded,L"%d",MinOsFree);
  1072. SecondDefined = TRUE;
  1073. }
  1074. }
  1075. }
  1076. if(!Fatal) {
  1077. ValidKeys[1] = ASCI_CR;
  1078. }
  1079. while(1) {
  1080. SpStartScreen(
  1081. Fatal ? SP_SCRN_WINNT_DRIVE_FULL_FATAL : SP_SCRN_WINNT_DRIVE_FULL,
  1082. 3,
  1083. HEADER_HEIGHT+1,
  1084. FALSE,
  1085. FALSE,
  1086. DEFAULT_ATTRIBUTE,
  1087. OsRgnDrive,
  1088. OsLoadFileName,
  1089. LoadIdentifier,
  1090. FirstDefined ? Drive1 : L"",
  1091. FirstDefined ? Drive1FreeNeeded : L"",
  1092. FirstDefined ? Drive1Free : L"",
  1093. SecondDefined ? Drive2 : L"",
  1094. SecondDefined ? Drive2FreeNeeded : L"",
  1095. SecondDefined ? Drive2Free : L""
  1096. );
  1097. SpDisplayStatusOptions(
  1098. DEFAULT_STATUS_ATTRIBUTE,
  1099. SP_STAT_F3_EQUALS_EXIT,
  1100. Fatal ? 0 : SP_STAT_ENTER_EQUALS_CONTINUE,
  1101. 0
  1102. );
  1103. if(SpWaitValidKey(ValidKeys,NULL,NULL) == KEY_F3) {
  1104. if(Fatal) {
  1105. SpDone(0,FALSE,TRUE);
  1106. } else {
  1107. SpConfirmExit();
  1108. }
  1109. } else {
  1110. //
  1111. // User hit CR in non-fatal case
  1112. //
  1113. return;
  1114. }
  1115. }
  1116. }
  1117. ENUMUPGRADETYPE
  1118. SppSelectNTToRepairByUpgrade(
  1119. OUT PSP_BOOT_ENTRY *BootEntryChosen
  1120. )
  1121. {
  1122. PVOID Menu;
  1123. ULONG MenuTopY;
  1124. ULONG ValidKeys[] = { KEY_F3,ASCI_ESC,0 };
  1125. ULONG Mnemonics[] = {MnemonicRepair,0 };
  1126. ULONG Keypress;
  1127. PSP_BOOT_ENTRY BootEntry,FirstUpgradeSet;
  1128. BOOL bDone;
  1129. ENUMUPGRADETYPE ret;
  1130. //
  1131. // Build up array of drive letters for all menu options
  1132. //
  1133. for(BootEntry = SpBootEntries; BootEntry != NULL; BootEntry = BootEntry->Next) {
  1134. if (BootEntry->Processable) {
  1135. SpGetUpgDriveLetter(
  1136. BootEntry->OsPartitionDiskRegion->DriveLetter,
  1137. BootEntry->DriveLetterString,
  1138. sizeof(DRIVELTR_STRING),
  1139. FALSE
  1140. );
  1141. }
  1142. }
  1143. bDone = FALSE;
  1144. while(!bDone) {
  1145. //
  1146. // Display the text that goes above the menu on the partitioning screen.
  1147. //
  1148. SpDisplayScreen(SP_SCRN_WINNT_REPAIR_BY_UPGRADE,3,CLIENT_TOP+1);
  1149. //
  1150. // Calculate menu placement. Leave one blank line
  1151. // and one line for a frame.
  1152. //
  1153. MenuTopY = NextMessageTopLine+2;
  1154. //
  1155. // Create a menu.
  1156. //
  1157. Menu = SpMnCreate(
  1158. MENU_LEFT_X,
  1159. MenuTopY,
  1160. MENU_WIDTH,
  1161. VideoVars.ScreenHeight-MenuTopY-2-STATUS_HEIGHT
  1162. );
  1163. ASSERT(Menu);
  1164. //
  1165. // Build up a menu of partitions and free spaces.
  1166. //
  1167. FirstUpgradeSet = NULL;
  1168. for(BootEntry = SpBootEntries; BootEntry != NULL; BootEntry = BootEntry->Next) {
  1169. if(BootEntry->Processable) {
  1170. swprintf(
  1171. TemporaryBuffer,
  1172. L"%ws:%ws %ws",
  1173. BootEntry->DriveLetterString,
  1174. BootEntry->OsDirectory,
  1175. BootEntry->FriendlyName
  1176. );
  1177. SpMnAddItem(
  1178. Menu,
  1179. TemporaryBuffer,
  1180. MENU_LEFT_X+MENU_INDENT,
  1181. MENU_WIDTH-(2*MENU_INDENT),
  1182. TRUE,
  1183. (ULONG_PTR)BootEntry
  1184. );
  1185. if(FirstUpgradeSet == NULL) {
  1186. FirstUpgradeSet = BootEntry;
  1187. }
  1188. }
  1189. }
  1190. //
  1191. // Initialize the status line.
  1192. //
  1193. SpDisplayStatusOptions(
  1194. DEFAULT_STATUS_ATTRIBUTE,
  1195. SP_STAT_F3_EQUALS_EXIT,
  1196. SP_STAT_R_EQUALS_REPAIR,
  1197. SP_STAT_ESC_EQUALS_NO_REPAIR,
  1198. 0
  1199. );
  1200. //
  1201. // Display the menu
  1202. //
  1203. SpMnDisplay(
  1204. Menu,
  1205. (ULONG_PTR)FirstUpgradeSet,
  1206. TRUE,
  1207. ValidKeys,
  1208. Mnemonics,
  1209. NULL,
  1210. &Keypress,
  1211. (PULONG_PTR)BootEntryChosen
  1212. );
  1213. //
  1214. // Now act on the user's selection.
  1215. //
  1216. switch(Keypress) {
  1217. case KEY_F3:
  1218. SpConfirmExit();
  1219. break;
  1220. case ASCI_ESC:
  1221. ret = DontUpgrade;
  1222. bDone = TRUE;
  1223. break;
  1224. default:
  1225. //
  1226. // Must be r=repair
  1227. //
  1228. ret = UpgradeFull;
  1229. bDone = TRUE;
  1230. break;
  1231. }
  1232. SpMnDestroy(Menu);
  1233. }
  1234. return(ret);
  1235. }
  1236. VOID
  1237. SpGetUpgDriveLetter(
  1238. IN WCHAR DriveLetter,
  1239. IN PWCHAR Buffer,
  1240. IN ULONG BufferSize,
  1241. IN BOOL AddColon
  1242. )
  1243. /*++
  1244. Routine Description:
  1245. This returns a unicode string containing the drive letter specified by
  1246. DriveLetter (if nonzero). If DriveLetter is 0, then we assume that we
  1247. are looking at a mirrored partition, and retrieve a localized string of
  1248. the form '(Mirror)'. If 'AddColon' is TRUE, then drive letters get a
  1249. colon appended (eg, "C:").
  1250. Arguments:
  1251. DriveLetter: Unicode drive letter, or 0 to denote a mirrored partition.
  1252. Buffer: Buffer to receive the unicode string
  1253. BufferSize: Size of the buffer.
  1254. AddColon: Boolean specifying whether colon should be added (has no
  1255. effect if DriveLetter is 0).
  1256. Returns:
  1257. Buffer contains the formatted Unicode string.
  1258. --*/
  1259. {
  1260. if(DriveLetter) {
  1261. if(BufferSize >= 2) {
  1262. *(Buffer++) = DriveLetter;
  1263. if(AddColon && BufferSize >= 3) {
  1264. *(Buffer++) = L':';
  1265. }
  1266. }
  1267. *Buffer = 0;
  1268. } else {
  1269. SpFormatMessage(Buffer, BufferSize, SP_UPG_MIRROR_DRIVELETTER);
  1270. }
  1271. }
  1272. BOOLEAN
  1273. SppWarnUpgradeWorkstationToServer(
  1274. IN ULONG MsgId
  1275. )
  1276. /*++
  1277. Routine Description:
  1278. Inform a user that that the installation that he/she selected to upgrade
  1279. is an NT Workstation, and that after the upgrade it will become a
  1280. Standard Server.
  1281. The user has the option to upgrade this or specify that he wants to
  1282. install Windows NT fresh.
  1283. Arguments:
  1284. MsgId - Screen to be displayed to the user.
  1285. Return Value:
  1286. BOOLEAN - Returns TRUE if the user wants to continue the upgrade, or
  1287. FALSE if the user wants to select another system to upgrade or
  1288. install fress.
  1289. --*/
  1290. {
  1291. ULONG ValidKeys[] = { ASCI_CR, ASCI_ESC, 0 };
  1292. ULONG c;
  1293. while(1) {
  1294. SpStartScreen(
  1295. MsgId,
  1296. 3,
  1297. HEADER_HEIGHT+1,
  1298. FALSE,
  1299. FALSE,
  1300. DEFAULT_ATTRIBUTE
  1301. );
  1302. SpDisplayStatusOptions(
  1303. DEFAULT_STATUS_ATTRIBUTE,
  1304. SP_STAT_ENTER_EQUALS_CONTINUE,
  1305. SP_STAT_ESC_EQUALS_CANCEL,
  1306. 0
  1307. );
  1308. switch(c=SpWaitValidKey(ValidKeys,NULL,NULL)) {
  1309. case KEY_F3:
  1310. SpConfirmExit();
  1311. break;
  1312. case ASCI_ESC:
  1313. return( FALSE );
  1314. case ASCI_CR:
  1315. return(TRUE);
  1316. default:
  1317. break;
  1318. }
  1319. }
  1320. }
  1321. VOID
  1322. SpCantFindBuildToUpgrade(
  1323. VOID
  1324. )
  1325. /*++
  1326. Routine Description:
  1327. Inform the user that we were unable to locate the build from which
  1328. he initiated unattended installation via winnt32.
  1329. This is a fatal condition.
  1330. Arguments:
  1331. None.
  1332. Return Value:
  1333. Does not return.
  1334. --*/
  1335. {
  1336. ULONG ValidKeys[2] = { KEY_F3, 0 };
  1337. CLEAR_CLIENT_SCREEN();
  1338. SpDisplayScreen(SP_SCRN_CANT_FIND_UPGRADE,3,HEADER_HEIGHT+1);
  1339. SpDisplayStatusOptions(DEFAULT_STATUS_ATTRIBUTE,SP_STAT_F3_EQUALS_EXIT,0);
  1340. SpWaitValidKey(ValidKeys,NULL,NULL);
  1341. SpDone(0,FALSE,FALSE);
  1342. }
  1343. void
  1344. SpUpgradeToNT50FileSystems(
  1345. PVOID SifHandle,
  1346. PDISK_REGION SystemPartitionRegion,
  1347. PDISK_REGION NTPartitionRegion,
  1348. PWSTR SetupSourceDevicePath,
  1349. PWSTR DirectoryOnSetupSource
  1350. )
  1351. /*++
  1352. Routine Description:
  1353. Perform any necessary upgrades of the file systems
  1354. for the NT40 to NT50 upgrade case.
  1355. Arguments:
  1356. SystemPartitionRegion - Pointer to the structure that describes the
  1357. system partition.
  1358. NTPartitionRegion - Pointer to the structure that describes the
  1359. NT partition.
  1360. SetupSourceDevicePath - NT device path where autochk.exe is located
  1361. DirectoryOnSourceDevice - Local source directory.
  1362. Return Value:
  1363. None.
  1364. --*/
  1365. {
  1366. NTSTATUS Status;
  1367. PDISK_REGION Region;
  1368. PUCHAR Win9xPath;
  1369. ULONG i,j,k;
  1370. PSP_BOOT_ENTRY BootEntry;
  1371. PWSTR NtPath;
  1372. BOOLEAN DoConvert = TRUE;
  1373. OBJECT_ATTRIBUTES Obja;
  1374. IO_STATUS_BLOCK IoStatusBlock;
  1375. PVOID ImageBase;
  1376. HANDLE SectionHandle;
  1377. ULONGLONG SourceVersion;
  1378. HANDLE SourceHandle;
  1379. UNICODE_STRING UnicodeString;
  1380. BOOLEAN IssueWarning = FALSE;
  1381. WCHAR SourceFile[MAX_PATH];
  1382. PWSTR MediaShortName;
  1383. PWSTR MediaDirectory;
  1384. UCHAR SysId;
  1385. #if defined(REMOTE_BOOT)
  1386. //
  1387. // If this is a remote boot installation, do not try to convert -- the
  1388. // NT partition in this case is on the remote boot server.
  1389. //
  1390. if (RemoteBootSetup && !RemoteInstallSetup) {
  1391. ConvertNtVolumeToNtfs = FALSE;
  1392. return;
  1393. }
  1394. #endif // defined(REMOTE_BOOT)
  1395. SpDetermineUniqueAndPresentBootEntries();
  1396. for(BootEntry = SpBootEntries; BootEntry != NULL; BootEntry = BootEntry->Next) {
  1397. if (!BootEntry->Processable) {
  1398. continue;
  1399. }
  1400. BootEntry->Processable = FALSE;
  1401. wcscpy( TemporaryBuffer, BootEntry->OsPartitionNtName );
  1402. SpConcatenatePaths( TemporaryBuffer, BootEntry->OsDirectory );
  1403. SpConcatenatePaths( TemporaryBuffer, L"\\system32\\ntoskrnl.exe" );
  1404. INIT_OBJA( &Obja, &UnicodeString, TemporaryBuffer );
  1405. Status = ZwCreateFile(
  1406. &SourceHandle,
  1407. FILE_GENERIC_READ,
  1408. &Obja,
  1409. &IoStatusBlock,
  1410. NULL,
  1411. 0,
  1412. 0,
  1413. FILE_OPEN,
  1414. FILE_NON_DIRECTORY_FILE | FILE_SYNCHRONOUS_IO_NONALERT,
  1415. NULL,
  1416. 0
  1417. );
  1418. if (NT_SUCCESS(Status)) {
  1419. Status = SpMapEntireFile( SourceHandle, &SectionHandle, &ImageBase, FALSE );
  1420. if (NT_SUCCESS(Status)) {
  1421. SpGetFileVersion( ImageBase, &BootEntry->KernelVersion );
  1422. BootEntry->Processable = TRUE;
  1423. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL,
  1424. "SETUP: SpUpgradeToNT50FileSystems: Kernel %p: NT%d.%d(Build %d) %d\n",
  1425. BootEntry,
  1426. (USHORT)(BootEntry->KernelVersion>>48),
  1427. (USHORT)(BootEntry->KernelVersion>>32),
  1428. (USHORT)(BootEntry->KernelVersion>>16),
  1429. (USHORT)(BootEntry->KernelVersion)));
  1430. SpUnmapFile( SectionHandle, ImageBase );
  1431. } else {
  1432. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "SETUP: SpUpgradeToNT50FileSystems() could not map kernel image [%ws], %lx\n", TemporaryBuffer, Status ));
  1433. }
  1434. ZwClose(SourceHandle);
  1435. } else {
  1436. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "SETUP: SpUpgradeToNT50FileSystems() corrupt boot config [%ws], %lx\n", TemporaryBuffer, Status ));
  1437. }
  1438. }
  1439. //
  1440. // count the number of "real" entries
  1441. //
  1442. k = 0;
  1443. for(BootEntry = SpBootEntries; BootEntry != NULL; BootEntry = BootEntry->Next) {
  1444. if (BootEntry->Processable) {
  1445. k += 1;
  1446. }
  1447. }
  1448. //
  1449. // check to see if a warning is necessary.
  1450. //
  1451. // If we're doing an upgrade, and there's only 1
  1452. // boot set, then we don't need a warning. However,
  1453. // if we're doing a clean install and there's at least
  1454. // 1 boot set, then warn (given the existing OS is
  1455. // old enough).
  1456. //
  1457. if( ( ((NTUpgrade == UpgradeFull) && (k > 1)) ||
  1458. ((NTUpgrade == DontUpgrade) && (k > 0)) ) &&
  1459. ( !UnattendedOperation ) && (!SpDrEnabled())) {
  1460. for(BootEntry = SpBootEntries; BootEntry != NULL && IssueWarning == FALSE; BootEntry = BootEntry->Next) {
  1461. if (!BootEntry->Processable || (BootEntry->KernelVersion == 0)) {
  1462. //
  1463. // bogus boot entry
  1464. //
  1465. } else if ((BootEntry->KernelVersion >> 48) < 4) {
  1466. IssueWarning = TRUE;
  1467. } else if ((BootEntry->KernelVersion >> 48) == 4 && (BootEntry->KernelVersion & 0xffff) <= 4) {
  1468. IssueWarning = TRUE;
  1469. }
  1470. }
  1471. }
  1472. // If there's any existing NT4.0 with servicepack less than 5 then warn.
  1473. if( k > 0) {
  1474. for(BootEntry = SpBootEntries; BootEntry != NULL && IssueWarning == FALSE; BootEntry = BootEntry->Next) {
  1475. if (!BootEntry->Processable || (BootEntry->KernelVersion == 0)) {
  1476. //
  1477. // bogus boot entry
  1478. //
  1479. } else if (BootEntry->MajorVersion == 4 && BootEntry->MinorVersion == 0 && BootEntry->BuildNumber == 1381 && BootEntry->ServicePack < 500) {
  1480. IssueWarning = TRUE;
  1481. }
  1482. }
  1483. }
  1484. if (IssueWarning) {
  1485. ULONG WarnKeys[] = { KEY_F3, 0 };
  1486. ULONG MnemonicKeys[] = { MnemonicContinueSetup, 0 };
  1487. while (IssueWarning) {
  1488. SpDisplayScreen(SP_SCRN_FSWARN,3,CLIENT_TOP+1);
  1489. SpDisplayStatusOptions(
  1490. DEFAULT_STATUS_ATTRIBUTE,
  1491. SP_STAT_C_EQUALS_CONTINUE_SETUP,
  1492. SP_STAT_F3_EQUALS_EXIT,
  1493. 0
  1494. );
  1495. SpInputDrain();
  1496. switch(SpWaitValidKey(WarnKeys,NULL,MnemonicKeys)) {
  1497. case KEY_F3:
  1498. SpConfirmExit();
  1499. return;
  1500. default:
  1501. IssueWarning = FALSE;
  1502. break;
  1503. }
  1504. }
  1505. }
  1506. #if 0
  1507. //
  1508. // now lets upgrade any nt40+sp3 ntfs file systems
  1509. //
  1510. MediaShortName = SpLookUpValueForFile(
  1511. SifHandle,
  1512. L"ntfs40.sys",
  1513. INDEX_WHICHMEDIA,
  1514. TRUE
  1515. );
  1516. SpGetSourceMediaInfo( SifHandle, MediaShortName, NULL, NULL, &MediaDirectory );
  1517. wcscpy( SourceFile, SetupSourceDevicePath );
  1518. SpConcatenatePaths( SourceFile, DirectoryOnSetupSource );
  1519. SpConcatenatePaths( SourceFile, MediaDirectory );
  1520. SpConcatenatePaths( SourceFile, L"ntfs40.sys" );
  1521. //
  1522. // Initialize the diamond decompression engine.
  1523. // This needs to be done, because SpCopyFileUsingNames() uses
  1524. // the decompression engine.
  1525. //
  1526. SpdInitialize();
  1527. for(BootEntry = SpBootEntries; BootEntry != NULL; BootEntry = BootEntry->Next) {
  1528. if (BootEntry->Processable &&
  1529. ((BootEntry->KernelVersion >> 48) == 4) &&
  1530. ((BootEntry->KernelVersion & 0xffff) == 4)) {
  1531. wcscpy( TemporaryBuffer, BootEntry->OsPartitionNtName );
  1532. SpConcatenatePaths( TemporaryBuffer, BootEntry->OsDirectory );
  1533. SpConcatenatePaths( TemporaryBuffer, L"\\system32\\drivers\\ntfs.sys" );
  1534. Status = SpCopyFileUsingNames(
  1535. SourceFile,
  1536. TemporaryBuffer,
  1537. 0,
  1538. COPY_NOVERSIONCHECK
  1539. );
  1540. if (!NT_SUCCESS(Status)) {
  1541. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "SETUP: SpUpgradeToNT50FileSystems() could not copy nt40 ntfs.sys driver, %lx\n", Status ));
  1542. }
  1543. SpMemFree(NtPath);
  1544. }
  1545. }
  1546. // Terminate diamond
  1547. SpdTerminate();
  1548. #endif
  1549. return;
  1550. }
  1551. BOOLEAN
  1552. SpDoBuildsMatch(
  1553. IN PVOID SifHandle,
  1554. ULONG TestBuildNum,
  1555. NT_PRODUCT_TYPE TestBuildType,
  1556. ULONG TestBuildSuiteMask,
  1557. BOOLEAN CurrentProductIsServer,
  1558. ULONG CurrentSuiteMask,
  1559. IN LCID LangId
  1560. )
  1561. /*++
  1562. Routine Description:
  1563. Checks if the current build the user is installing matches the build we're
  1564. checking against.
  1565. We check:
  1566. 1. do the build numbers match?
  1567. 2. do the build types match? (nt server and nt professional don't match)
  1568. 3. do the product suites match? (nt advanced server vs. data center)
  1569. Arguments:
  1570. SifHandle - Handle to txtsetup.sif to find the source language
  1571. TestBuildNum - The build number of the build we're checking against
  1572. TestBuildType - The type of build we're checking against
  1573. TestBuildSuiteMask - Type of product suite as mask we're checking against
  1574. CurrentProductIsServer - If TRUE, the current build is NT Server
  1575. CurrentSuiteMask - Type of suite mask
  1576. LangId - System Language Id of the installation to check. If -1, Lang Id
  1577. check is ignored.
  1578. Return Value:
  1579. BOOLEAN - Returns TRUE if the builds match.
  1580. --*/
  1581. {
  1582. #define PRODUCTSUITES_TO_MATCH (( VER_SUITE_SMALLBUSINESS \
  1583. | VER_SUITE_ENTERPRISE \
  1584. | VER_SUITE_BACKOFFICE \
  1585. | VER_SUITE_COMMUNICATIONS \
  1586. | VER_SUITE_SMALLBUSINESS_RESTRICTED \
  1587. | VER_SUITE_EMBEDDEDNT \
  1588. | VER_SUITE_DATACENTER \
  1589. | VER_SUITE_PERSONAL))
  1590. BOOLEAN retval;
  1591. LANGID DefLangId = -1;
  1592. DWORD Version = 0, BuildNumber = 0;
  1593. if (!DetermineSourceVersionInfo(&Version, &BuildNumber)) {
  1594. retval = FALSE;
  1595. goto exit;
  1596. }
  1597. if(TestBuildNum != BuildNumber) {
  1598. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL,
  1599. "SETUP: SpDoBuildsMatch() has build mismatch, %d != %d\n",
  1600. TestBuildNum, BuildNumber));
  1601. retval = FALSE;
  1602. goto exit;
  1603. }
  1604. //
  1605. // build number test passed. now check for server vs. professional
  1606. //
  1607. if (((TestBuildType == NtProductWinNt) && CurrentProductIsServer) ||
  1608. ((TestBuildType == NtProductServer) && !CurrentProductIsServer)) {
  1609. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL,
  1610. "SETUP: SpDoBuildsMatch() has server/professional mismatch\n" ));
  1611. retval = FALSE;
  1612. goto exit;
  1613. }
  1614. //
  1615. // now check product suites.
  1616. // note that we don't check for all product suites, only
  1617. // suites that have their own SKU
  1618. //
  1619. if ((CurrentSuiteMask & PRODUCTSUITES_TO_MATCH) != (TestBuildSuiteMask & PRODUCTSUITES_TO_MATCH)) {
  1620. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL,
  1621. "SETUP: SpDoBuildsMatch() has suite mismatch (dest = %x) (source = %x\n",
  1622. TestBuildSuiteMask,CurrentSuiteMask ));
  1623. retval = FALSE;
  1624. goto exit;
  1625. }
  1626. //
  1627. // language IDs should also match (if requested)
  1628. //
  1629. if (LangId != -1) {
  1630. PWSTR LangIdStr = SpGetSectionKeyIndex(SifHandle, SIF_NLS, SIF_DEFAULTLAYOUT, 0);
  1631. PWSTR EndChar;
  1632. if (LangIdStr)
  1633. DefLangId = (LANGID)SpStringToLong(LangIdStr, &EndChar, 16);
  1634. //
  1635. // note : currently we are only interested in primary language IDs
  1636. //
  1637. retval = (PRIMARYLANGID(DefLangId) == PRIMARYLANGID(LangId)) ? TRUE : FALSE;
  1638. goto exit;
  1639. }
  1640. retval = TRUE;
  1641. exit:
  1642. return(retval);
  1643. }
  1644. NTSTATUS
  1645. SpGetMediaDetails(
  1646. IN PWSTR CdInfDirPath,
  1647. OUT PCCMEDIA MediaObj
  1648. )
  1649. /*++
  1650. Routine Description:
  1651. Gets the details of the current CD in CCMEDIA structure
  1652. Arguments:
  1653. CdInfDirPath - The path to the inf directories on CD-ROM drive
  1654. MediaObj - The pointer to the media object in which the details
  1655. are returned
  1656. Return Value:
  1657. Returns STATUS_SUCCESS if success otherwise appropriate status error code.
  1658. --*/
  1659. {
  1660. NTSTATUS Status = STATUS_INVALID_PARAMETER;
  1661. WCHAR DosNetPath[MAX_PATH];
  1662. WCHAR SetuppIniPath[MAX_PATH];
  1663. PVOID SetuppIniHandle = 0;
  1664. PVOID DosNetHandle = 0;
  1665. ULONG ErrorLine = 0;
  1666. WCHAR Pid[32] = {0};
  1667. WCHAR PidData[256] = {0};
  1668. PWSTR TempPtr;
  1669. BOOLEAN UpgradeMode = FALSE;
  1670. DWORD Type = COMPLIANCE_INSTALLTYPE_UNKNOWN;
  1671. DWORD Variation = COMPLIANCE_INSTALLVAR_UNKNOWN;
  1672. DWORD Version = 0, BuildNumber = 0;
  1673. if (CdInfDirPath && MediaObj) {
  1674. BOOLEAN VersionDetected = FALSE;
  1675. wcscpy(DosNetPath, CdInfDirPath);
  1676. wcscpy(SetuppIniPath, CdInfDirPath);
  1677. SpConcatenatePaths(DosNetPath, L"dosnet.inf");
  1678. SpConcatenatePaths(SetuppIniPath, L"setupp.ini");
  1679. //
  1680. // load setupp.ini file and parse it
  1681. //
  1682. Status = SpLoadSetupTextFile(
  1683. SetuppIniPath,
  1684. NULL, // No image already in memory
  1685. 0, // Image size is empty
  1686. &SetuppIniHandle,
  1687. &ErrorLine,
  1688. TRUE,
  1689. FALSE
  1690. );
  1691. if(NT_SUCCESS(Status)) {
  1692. Status = STATUS_FILE_INVALID;
  1693. //
  1694. // get the PID
  1695. //
  1696. TempPtr = SpGetSectionKeyIndex(SetuppIniHandle, L"Pid", L"Pid", 0);
  1697. if (TempPtr) {
  1698. wcscpy(Pid, TempPtr);
  1699. //
  1700. // get PID ExtraData
  1701. //
  1702. TempPtr = SpGetSectionKeyIndex(SetuppIniHandle, L"Pid", L"ExtraData", 0);
  1703. if (TempPtr) {
  1704. wcscpy(PidData, TempPtr);
  1705. //
  1706. // Get stepup mode & install variation based on PID
  1707. //
  1708. if (SpGetStepUpMode(PidData, &UpgradeMode) &&
  1709. pSpGetCurrentInstallVariation(Pid, &Variation)) {
  1710. Status = STATUS_SUCCESS;
  1711. } else {
  1712. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "SpGetMediaDetails: Could "
  1713. "not find StepUp mode or variation of install CD\n"));
  1714. }
  1715. } else {
  1716. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "SpGetMediaDetails: Could not get "
  1717. "PidExtraData from Setupp.ini\n"));
  1718. }
  1719. } else {
  1720. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "SpGetMediaDetails: Could not get Pid from Setupp.ini\n"));
  1721. }
  1722. } else {
  1723. //
  1724. // Silently fail if unable to read setupp.ini
  1725. //
  1726. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_INFO_LEVEL, "SpGetMediaDetails: Unable to read setupp.ini. "
  1727. "Status = %lx \n", Status));
  1728. }
  1729. if (SetuppIniHandle)
  1730. SpFreeTextFile(SetuppIniHandle);
  1731. if (NT_SUCCESS(Status)) {
  1732. //
  1733. // load and parse dosnet.inf
  1734. //
  1735. Status = SpLoadSetupTextFile(
  1736. DosNetPath,
  1737. NULL, // No image already in memory
  1738. 0, // Image size is empty
  1739. &DosNetHandle,
  1740. &ErrorLine,
  1741. TRUE,
  1742. FALSE
  1743. );
  1744. if (NT_SUCCESS(Status)) {
  1745. Status = STATUS_FILE_INVALID;
  1746. //
  1747. // get ProductType from Miscellaneous section
  1748. //
  1749. TempPtr = SpGetSectionKeyIndex(DosNetHandle, L"Miscellaneous",
  1750. L"ProductType", 0);
  1751. if (TempPtr) {
  1752. UNICODE_STRING UnicodeStr;
  1753. ULONG Value = -1;
  1754. RtlInitUnicodeString(&UnicodeStr, TempPtr);
  1755. Status = RtlUnicodeStringToInteger(&UnicodeStr, 10, &Value);
  1756. switch (Value) {
  1757. case 0:
  1758. Type = COMPLIANCE_INSTALLTYPE_NTW;
  1759. break;
  1760. case 1:
  1761. Type = COMPLIANCE_INSTALLTYPE_NTS;
  1762. break;
  1763. case 2:
  1764. Type = COMPLIANCE_INSTALLTYPE_NTSE;
  1765. break;
  1766. case 3:
  1767. Type = COMPLIANCE_INSTALLTYPE_NTSDTC;
  1768. break;
  1769. case 4:
  1770. Type = COMPLIANCE_INSTALLTYPE_NTWP;
  1771. break;
  1772. default:
  1773. break;
  1774. }
  1775. //
  1776. // Get the version also off from driverver in dosnet.inf
  1777. //
  1778. TempPtr = SpGetSectionKeyIndex(DosNetHandle, L"Version",
  1779. L"DriverVer", 1);
  1780. if (TempPtr) {
  1781. if (NT_SUCCESS(SpGetVersionFromStr(TempPtr,
  1782. &Version, &BuildNumber))) {
  1783. VersionDetected = TRUE;
  1784. }
  1785. Status = STATUS_SUCCESS;
  1786. }
  1787. if (Type != COMPLIANCE_INSTALLTYPE_UNKNOWN) {
  1788. Status = STATUS_SUCCESS;
  1789. } else {
  1790. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "SpGetMediaDetails: Could not get product type"
  1791. " from dosnet.inf\n"));
  1792. }
  1793. } else {
  1794. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "SpGetMediaDetails: "
  1795. "Could not get ProductType from dosnet.inf\n"));
  1796. }
  1797. } else {
  1798. //
  1799. // Silently fail if unable to read dosnet.inf
  1800. //
  1801. KdPrintEx((DPFLTR_SETUP_ID, DPFLTR_ERROR_LEVEL, "SpGetMediaDetails: Unable to read dosnet.inf. "
  1802. "Status = %lx \n", Status ));
  1803. }
  1804. }
  1805. if (DosNetHandle)
  1806. SpFreeTextFile(DosNetHandle);
  1807. //
  1808. // Fall back to old way of finding version, if we failed
  1809. // to get one from dosnet.inf
  1810. //
  1811. if (NT_SUCCESS(Status) && !VersionDetected) {
  1812. if (!DetermineSourceVersionInfo(&Version, &BuildNumber))
  1813. Status = STATUS_FILE_INVALID;
  1814. }
  1815. //
  1816. // fill in the media details
  1817. //
  1818. if (NT_SUCCESS(Status) &&
  1819. ! CCMediaInitialize(MediaObj, Type, Variation, UpgradeMode, Version, BuildNumber)) {
  1820. Status = STATUS_FILE_INVALID;
  1821. }
  1822. }
  1823. return Status;
  1824. }