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.

861 lines
21 KiB

  1. /*++
  2. Copyright (c) 1993-1999 Microsoft Corporation
  3. Module Name:
  4. syspart.c
  5. Abstract:
  6. Routines to determine the system partition on x86 machines.
  7. Author:
  8. Ted Miller (tedm) 30-June-1994
  9. Revision History:
  10. Copied from winnt32 to risetup by ChuckL 12-May-1999
  11. --*/
  12. #include "pch.h"
  13. #pragma hdrstop
  14. #include <winioctl.h>
  15. DEFINE_MODULE("SysPart");
  16. #define MALLOC(_size) TraceAlloc(LPTR,(_size))
  17. #define FREE(_p) TraceFree(_p)
  18. UINT
  19. MyGetDriveType(
  20. IN TCHAR Drive
  21. );
  22. BOOL
  23. IsNEC98(
  24. VOID
  25. );
  26. #define WINNT_DONT_MATCH_PARTITION 0
  27. #define WINNT_MATCH_PARTITION_NUMBER 1
  28. #define WINNT_MATCH_PARTITION_STARTING_OFFSET 2
  29. DWORD
  30. FindSystemPartitionSignature(
  31. IN LPCTSTR AdapterKeyName,
  32. OUT LPTSTR Signature
  33. );
  34. DWORD
  35. GetSystemVolumeGUID(
  36. IN LPTSTR Signature,
  37. OUT LPTSTR SysVolGuid
  38. );
  39. BOOL
  40. DoDiskSignaturesCompare(
  41. IN LPCTSTR Signature,
  42. IN LPCTSTR DriveName,
  43. IN OUT PVOID Compare,
  44. IN DWORD Action
  45. );
  46. BOOL
  47. x86DetermineSystemPartition(
  48. IN HWND ParentWindow,
  49. OUT PTCHAR SysPartDrive
  50. )
  51. /*++
  52. Routine Description:
  53. Determine the system partition on x86 machines.
  54. On Win95, we always return C:. For NT, read on.
  55. The system partition is the primary partition on the boot disk.
  56. Usually this is the active partition on disk 0 and usually it's C:.
  57. However the user could have remapped drive letters and generally
  58. determining the system partition with 100% accuracy is not possible.
  59. With there being differences in the IO system mapping and introduction of Volumes for NT 50
  60. this has now become complicated. Listed below are the algorithms
  61. NT 5.0 Beta 2 and above :
  62. 1. Get the signature from the registry. Located at
  63. HKLM\Hardware\Description\System\<MultifunctionAdapter or EisaAdapter>\<some Bus No.>\DiskController\0\DiskPeripheral\0\Identifier
  64. 2. Go Through all of the volumes in the system with FindFirstVolume/FindNextVolume/FindVolumeClose.
  65. 3. Take off the trailing backslash to the name returne to get \\?\Volume{guid}.
  66. 4. IOCTL_STORAGE_GET_DEVICE_NUMBER with \\?\Volume{guid} => Check for FILE_DEVICE_DISK. Remember the partition number. Goto 6
  67. 5. If IOCTL_STORAGE_GET_DEVICE_NUMBER fails then use IOCTL_VOLUME_GET_VOLUME_DISK_EXTENTS which returns a list of harddisks.
  68. For each harddisk remember the starting offset and goto 6.
  69. 6. Check Harddisk # by using \\.\PhysicalDrive# with IOCTL_DISK_GET_DRIVE_LAYOUT. If the signature matches then this is the disk we boot from.
  70. 7. To find the partition that we boot from we look for boot indicator. If we used 2 we try to match the partition number stored in 6
  71. else if 3 we try to match the starting offset.Then you have a \\?\Volume{guid}\ name for the SYSTEM volume.
  72. 8. Call GetVolumeNameForVolumeMountPoint with A:\, B:\, C:\, ... and check the result of the form \\?\Volume{guid}\ against your match
  73. to see what the drive letter is.
  74. Important: Since the *Volume* APIs are post Beta2 we do dynamic loading of kernel32.dll based on the build number returned.
  75. Versions below NT 5.0 Beta 2
  76. 1. Get the signature from the registry. Located at
  77. HKLM\Hardware\Description\System\<MultifunctionAdapter or EisaAdapter>\<some Bus No.>\DiskController\0\DiskPeripheral\0\Identifier
  78. 2. Enumerate the \?? directory and look for all entries that begin with PhysicalDrive#.
  79. 3. For each of the Disks look for a match with the signature above and if they match then find out the partition number used to boot
  80. using IOCTL_DISK_GET_DRIVE_LAYOUT and the BootIndicator bit.
  81. 4. On finding the Boot partition create a name of the form \Device\Harddisk#\Partition#
  82. 5. Then go through c:,d:...,z: calling QueryDosDeviceName and look for a match. That would be your system partition drive letter
  83. Arguments:
  84. ParentWindow - supplies window handle for window to be the parent for
  85. any dialogs, etc.
  86. SysPartDrive - if successful, receives drive letter of system partition.
  87. Return Value:
  88. Boolean value indicating whether SysPartDrive has been filled in.
  89. If FALSE, the user will have been infomred about why.
  90. --*/
  91. {
  92. TCHAR DriveName[4];
  93. BOOL GotIt=FALSE;
  94. TCHAR Buffer[512];
  95. TCHAR Drive;
  96. BOOL b;
  97. TCHAR SysPartSig[20], PartitionNum[MAX_PATH], SysVolGuid[MAX_PATH];
  98. TCHAR DriveVolGuid[MAX_PATH];
  99. if(IsNEC98()) {
  100. if (!GetWindowsDirectory(Buffer,sizeof(Buffer)/sizeof(TCHAR))) {
  101. return(FALSE);
  102. }
  103. *SysPartDrive = Buffer[0];
  104. return(TRUE);
  105. }
  106. //Get signature from registry - Step 1 listed above
  107. if( (FindSystemPartitionSignature(TEXT("Hardware\\Description\\System\\EisaAdapter"),SysPartSig) != ERROR_SUCCESS )
  108. && (FindSystemPartitionSignature(TEXT("Hardware\\Description\\System\\MultiFunctionAdapter"),SysPartSig) != ERROR_SUCCESS ) ){
  109. GotIt = FALSE;
  110. goto c0;
  111. }
  112. //Get the SystemVolumeGUID - steps 2 through 7 listed above ( Beta 2 and after )
  113. if( GetSystemVolumeGUID( SysPartSig, SysVolGuid ) != ERROR_SUCCESS ){
  114. GotIt = FALSE;
  115. goto c0;
  116. }
  117. DriveName[1] = TEXT(':');
  118. //
  119. // Enumerate all drive letters and compare their device names
  120. //
  121. for(Drive=TEXT('A'); Drive<=TEXT('Z'); Drive++) {
  122. if(MyGetDriveType(Drive) == DRIVE_FIXED) {
  123. DriveName[0] = Drive;
  124. DriveName[2] = '\\';
  125. DriveName[3] = 0;
  126. if((*GetVolumeNameForVolumeMountPoint)((LPWSTR)DriveName, (LPWSTR)DriveVolGuid, MAX_PATH*sizeof(TCHAR))){
  127. if(!lstrcmp(DriveVolGuid, SysVolGuid) ){
  128. GotIt = TRUE; // Found it
  129. break;
  130. }
  131. }
  132. }
  133. }
  134. // This helps for some builds ~1500 < buildnum < 1877 that are in a tough spot
  135. if(!GotIt) {
  136. //
  137. // Strange case, just assume C:
  138. //
  139. GotIt = TRUE;
  140. Drive = TEXT('C');
  141. }
  142. c0:
  143. if(GotIt) {
  144. *SysPartDrive = Drive;
  145. }
  146. return(GotIt);
  147. }
  148. DWORD
  149. GetSystemVolumeGUID(
  150. IN LPTSTR Signature,
  151. OUT LPTSTR SysVolGuid
  152. )
  153. /*++
  154. Routine Description:
  155. This routine enumerates all the volumes and if successful returns the \\?\Volume{guid} name for the system partition.
  156. Arguments:
  157. Signature - supplies a disk signature of the Boot disk so that it can be compared against. The details
  158. to getting this value are detailed in the comments for x86DetermineSystemPartition.
  159. SysVolGuid - If successful, will contain a name of form \\?\Volume{guid} for the System Partition (the one we use to boot)
  160. Return Value:
  161. Returns NO_ERROR if successful, otherwise it contains the error code.
  162. --*/
  163. {
  164. HANDLE hVolume, h;
  165. TCHAR VolumeName[MAX_PATH];
  166. TCHAR driveName[30];
  167. BOOL b,ret, DoExtent, MatchFound;
  168. STORAGE_DEVICE_NUMBER number;
  169. DWORD Err = NO_ERROR;
  170. DWORD cnt;
  171. PVOLUME_DISK_EXTENTS Extent = NULL;
  172. PDISK_EXTENT Start, i;
  173. DWORD ExtentSize, bytes;
  174. PVOID p;
  175. ULONG PartitionNumToCompare;
  176. LARGE_INTEGER StartingOffToCompare;
  177. //Enuberate all volumes
  178. hVolume = (*FindFirstVolume)( (LPWSTR)VolumeName, MAX_PATH );
  179. if( hVolume == INVALID_HANDLE_VALUE ){
  180. return GetLastError();
  181. }
  182. MatchFound = FALSE;
  183. do{
  184. //Remove trailing backslash
  185. DoExtent = FALSE;
  186. if( wcsrchr( VolumeName,TEXT('\\') ) ){
  187. *wcsrchr( VolumeName,TEXT('\\') ) = 0;
  188. }else{
  189. continue;
  190. }
  191. //Open the volume
  192. h = CreateFile(VolumeName, GENERIC_READ, FILE_SHARE_READ |
  193. FILE_SHARE_WRITE, NULL, OPEN_EXISTING,
  194. FILE_ATTRIBUTE_NORMAL, INVALID_HANDLE_VALUE);
  195. if (h == INVALID_HANDLE_VALUE) {
  196. continue; // Move on to next volume
  197. }
  198. //Get the disk number
  199. ret = DeviceIoControl(h, IOCTL_STORAGE_GET_DEVICE_NUMBER, NULL, 0,
  200. &number, sizeof(number), &bytes, NULL);
  201. if( !ret ){
  202. // Try using IOCTL_VOLUME_GET_VOLUME_DISK_EXTENTS if the above failed
  203. Extent = (PVOLUME_DISK_EXTENTS)MALLOC(1024);
  204. ExtentSize = 1024;
  205. if(!Extent) {
  206. CloseHandle( h );
  207. Err = ERROR_NOT_ENOUGH_MEMORY;
  208. goto cleanup;
  209. }
  210. retry:
  211. ret = DeviceIoControl( h, IOCTL_VOLUME_GET_VOLUME_DISK_EXTENTS,
  212. NULL,0,(PVOID)Extent,ExtentSize,&bytes,NULL);
  213. if(!ret) {
  214. if((Err=GetLastError()) == ERROR_INSUFFICIENT_BUFFER) {
  215. ExtentSize += 1024;
  216. FREE(Extent);
  217. if(Extent = (PVOLUME_DISK_EXTENTS)MALLOC(ExtentSize)) {
  218. ;
  219. } else {
  220. CloseHandle( h );
  221. Err = ERROR_NOT_ENOUGH_MEMORY;
  222. goto cleanup;
  223. }
  224. goto retry;
  225. } else {
  226. CloseHandle( h );
  227. continue;
  228. }
  229. }else{
  230. DoExtent = TRUE;
  231. }
  232. }
  233. // Done with the handle this time around
  234. CloseHandle( h );
  235. if( !DoExtent ){
  236. //
  237. // Check to see if this is a disk and not CDROM etc.
  238. //
  239. if( number.DeviceType == FILE_DEVICE_DISK ){
  240. // Remember the partition number
  241. PartitionNumToCompare = number.PartitionNumber;
  242. wsprintf( driveName, TEXT("\\\\.\\PhysicalDrive%lu"), number.DeviceNumber );
  243. if(DoDiskSignaturesCompare( Signature, driveName, (PVOID)&PartitionNumToCompare, WINNT_MATCH_PARTITION_NUMBER ) ){
  244. MatchFound = TRUE;
  245. Err = NO_ERROR;
  246. lstrcpy( SysVolGuid, VolumeName );
  247. SysVolGuid[lstrlen(VolumeName)]=TEXT('\\');
  248. SysVolGuid[lstrlen(VolumeName)+1]=0;
  249. break;
  250. }
  251. }
  252. // Move on ..
  253. continue;
  254. }else{
  255. // Go through all disks and try for match
  256. Start = Extent->Extents;
  257. cnt = 0;
  258. for( i = Start; cnt < Extent->NumberOfDiskExtents; i++ ){
  259. cnt++;
  260. // Remember the starting offset
  261. StartingOffToCompare = i->StartingOffset;
  262. wsprintf( driveName, TEXT("\\\\.\\PhysicalDrive%lu"), i->DiskNumber );
  263. if(DoDiskSignaturesCompare( Signature, driveName, (PVOID)&StartingOffToCompare, WINNT_MATCH_PARTITION_STARTING_OFFSET ) ){
  264. MatchFound = TRUE;
  265. Err = NO_ERROR;
  266. lstrcpy( SysVolGuid, VolumeName );
  267. SysVolGuid[lstrlen(VolumeName)]=TEXT('\\');
  268. SysVolGuid[lstrlen(VolumeName)+1]=0;
  269. break;
  270. }
  271. }
  272. }
  273. if( MatchFound )
  274. break;
  275. }while( (*FindNextVolume)( hVolume, (LPWSTR)VolumeName, MAX_PATH ));
  276. cleanup:
  277. if( hVolume != INVALID_HANDLE_VALUE )
  278. (*FindVolumeClose)( hVolume );
  279. if( Extent != NULL ) {
  280. FREE(Extent);
  281. }
  282. return Err;
  283. }
  284. BOOL
  285. DoDiskSignaturesCompare(
  286. IN LPCTSTR Signature,
  287. IN LPCTSTR DriveName,
  288. IN OUT PVOID Compare,
  289. IN DWORD Action
  290. )
  291. /*++
  292. Routine Description:
  293. This routine compares the given disk signature with the one for the specified physical disk.
  294. Arguments:
  295. Signature - supplies a disk signature of the Boot disk so that it can be compared against. The details
  296. to getting this value are detailed in the comments for x86DetermineSystemPartition.
  297. DriveName - Physical Drive name of the form \\.\PhysicalDrive#
  298. Compare - A pointer to a storage variable. The type depends on the value of Action
  299. Action - Should be one of the following
  300. WINNT_DONT_MATCH_PARTITION - Once disk signatures match it returns the boot partition number in Compare. Compare should be a PULONG.
  301. WINNT_MATCH_PARTITION_NUMBER - Once disk signatures match it tries to match the boot partition number with the number passed in
  302. through Compare. Compare should be PULONG.
  303. WINNT_MATCH_PARTITION_STARTING_OFFSET - Once disk signatures match it tries to match the boot partition starting offset with the
  304. starting offset number passed in through Compare. Compare should be PLARGE_INTEGER.
  305. Return Value:
  306. Returns TRUE if successful in getting a match.
  307. --*/
  308. {
  309. TCHAR temp[30];
  310. BOOL b,Found = FALSE;
  311. PLARGE_INTEGER Starting_Off;
  312. PPARTITION_INFORMATION Start, i;
  313. HANDLE hDisk;
  314. PDRIVE_LAYOUT_INFORMATION DriveLayout = NULL;
  315. DWORD DriveLayoutSize;
  316. DWORD cnt;
  317. DWORD DataSize;
  318. PULONG Disk_Num;
  319. ULONG Sig;
  320. if(!Compare )
  321. return FALSE;
  322. if( (Action==WINNT_MATCH_PARTITION_STARTING_OFFSET) && Compare )
  323. Starting_Off = (PLARGE_INTEGER) Compare;
  324. else
  325. Disk_Num = (PULONG) Compare;
  326. // On any failure return FALSE
  327. //
  328. // Get drive layout info for this physical disk.
  329. // If we can't do this something is very wrong.
  330. //
  331. hDisk = CreateFile(
  332. DriveName,
  333. FILE_READ_ATTRIBUTES | FILE_READ_DATA,
  334. FILE_SHARE_READ | FILE_SHARE_WRITE,
  335. NULL,
  336. OPEN_EXISTING,
  337. 0,
  338. NULL
  339. );
  340. if(hDisk == INVALID_HANDLE_VALUE) {
  341. return FALSE;
  342. }
  343. //
  344. // Get partition information.
  345. //
  346. DriveLayout = (PDRIVE_LAYOUT_INFORMATION)MALLOC(1024);
  347. DriveLayoutSize = 1024;
  348. if(!DriveLayout) {
  349. goto cleanexit;
  350. }
  351. retry:
  352. b = DeviceIoControl(
  353. hDisk,
  354. IOCTL_DISK_GET_DRIVE_LAYOUT,
  355. NULL,
  356. 0,
  357. (PVOID)DriveLayout,
  358. DriveLayoutSize,
  359. &DataSize,
  360. NULL
  361. );
  362. if(!b) {
  363. if(GetLastError() == ERROR_INSUFFICIENT_BUFFER) {
  364. DriveLayoutSize += 1024;
  365. FREE(DriveLayout);
  366. if(DriveLayout = (PDRIVE_LAYOUT_INFORMATION)MALLOC(DriveLayoutSize)) {
  367. ;
  368. } else {
  369. goto cleanexit;
  370. }
  371. goto retry;
  372. } else {
  373. goto cleanexit;
  374. }
  375. }else{
  376. // Now walk the Drive_Layout to find the boot partition
  377. Start = DriveLayout->PartitionEntry;
  378. cnt = 0;
  379. /*
  380. _ultot( DriveLayout->Signature, temp, 16 );
  381. if( lstrcmpi( temp, Signature ) )
  382. goto cleanexit;
  383. */
  384. Sig = wcstoul( Signature, NULL, 16 );
  385. if( Sig != DriveLayout->Signature )
  386. goto cleanexit;
  387. for( i = Start; cnt < DriveLayout->PartitionCount; i++ ){
  388. cnt++;
  389. if( i->BootIndicator == TRUE ){
  390. if( Action == WINNT_DONT_MATCH_PARTITION ){
  391. *Disk_Num = i->PartitionNumber;
  392. Found = TRUE;
  393. goto cleanexit;
  394. }
  395. if( Action == WINNT_MATCH_PARTITION_NUMBER ){
  396. if( *Disk_Num == i->PartitionNumber ){
  397. Found = TRUE;
  398. goto cleanexit;
  399. }
  400. }else{
  401. if( Starting_Off->QuadPart == i->StartingOffset.QuadPart ){
  402. Found = TRUE;
  403. goto cleanexit;
  404. }
  405. }
  406. break;
  407. }
  408. }
  409. }
  410. cleanexit:
  411. if( hDisk != INVALID_HANDLE_VALUE )
  412. CloseHandle( hDisk );
  413. if( DriveLayout != NULL ) {
  414. FREE(DriveLayout);
  415. }
  416. return Found;
  417. }
  418. DWORD
  419. FindSystemPartitionSignature(
  420. IN LPCTSTR AdapterKeyName,
  421. OUT LPTSTR Signature
  422. )
  423. /*++
  424. Routine Description:
  425. This routine fetches the disk signature for the first disk that the BIOS sees. This has to be the disk that we boot from on x86s.
  426. It is at location <AdapterKeyName>\<some Bus No.>\DiskController\0\DiskPeripheral\0\Identifier
  427. Arguments:
  428. Signature - If successful will contain the signature of the disk we boot off from in Hex.
  429. Return Value:
  430. Returns ERROR_SUCCESS if successful, otherwise it contains the error code.
  431. --*/
  432. {
  433. DWORD Err, dwSize;
  434. HKEY hkey, BusKey, Controller, SystemDiskKey;
  435. int busnumber;
  436. TCHAR BusString[20], Identifier[100];
  437. Err = RegOpenKeyEx( HKEY_LOCAL_MACHINE,
  438. AdapterKeyName,
  439. 0,
  440. KEY_READ,
  441. &hkey );
  442. if( Err != ERROR_SUCCESS )
  443. return Err;
  444. // Start enumerating the buses
  445. for( busnumber=0; ;busnumber++){
  446. wsprintf( BusString, TEXT("%d"), busnumber );
  447. Err = RegOpenKeyEx( hkey,
  448. BusString,
  449. 0,
  450. KEY_READ,
  451. &BusKey );
  452. if( Err != ERROR_SUCCESS ){
  453. RegCloseKey( hkey );
  454. return Err;
  455. }
  456. Err = RegOpenKeyEx( BusKey,
  457. TEXT("DiskController"),
  458. 0,
  459. KEY_READ,
  460. &Controller );
  461. RegCloseKey(BusKey); // Not needed anymore
  462. if( Err != ERROR_SUCCESS ) // Move on to next bus
  463. continue;
  464. RegCloseKey( hkey ); // Not needed anymore
  465. Err = RegOpenKeyEx( Controller,
  466. TEXT("0\\DiskPeripheral\\0"),
  467. 0,
  468. KEY_READ,
  469. &SystemDiskKey );
  470. if( Err != ERROR_SUCCESS ){
  471. RegCloseKey( Controller );
  472. return Err;
  473. }
  474. RegCloseKey( Controller ); // Not needed anymore
  475. dwSize = sizeof(Identifier);
  476. Err = RegQueryValueEx( SystemDiskKey,
  477. TEXT("Identifier"),
  478. NULL,
  479. NULL,
  480. (PBYTE) Identifier,
  481. &dwSize);
  482. if( Err != ERROR_SUCCESS ){
  483. RegCloseKey( SystemDiskKey );
  484. return Err;
  485. }
  486. if( Identifier && (lstrlen(Identifier) > 9 ) ){
  487. PWCHAR p;
  488. lstrcpy( Signature,Identifier+9);
  489. p = wcsrchr( Signature,TEXT('-') );
  490. if( p ) {
  491. *p = 0;
  492. }
  493. RegCloseKey( SystemDiskKey );
  494. return ERROR_SUCCESS;
  495. }
  496. else{
  497. RegCloseKey( SystemDiskKey );
  498. return Err;
  499. }
  500. }
  501. // Should never get here
  502. RegCloseKey( hkey );
  503. return ERROR_PATH_NOT_FOUND;
  504. }
  505. UINT
  506. MyGetDriveType(
  507. IN TCHAR Drive
  508. )
  509. /*++
  510. Routine Description:
  511. Same as GetDriveType() Win32 API except on NT returns
  512. DRIVE_FIXED for removeable hard drives.
  513. Arguments:
  514. Drive - supplies drive letter whose type is desired.
  515. Return Value:
  516. Same as GetDriveType().
  517. --*/
  518. {
  519. TCHAR DriveNameNt[] = TEXT("\\\\.\\?:");
  520. TCHAR DriveName[] = TEXT("?:\\");
  521. HANDLE hDisk;
  522. BOOL b;
  523. UINT rc;
  524. DWORD DataSize;
  525. DISK_GEOMETRY MediaInfo;
  526. //
  527. // First, get the win32 drive type. If it tells us DRIVE_REMOVABLE,
  528. // then we need to see whether it's a floppy or hard disk. Otherwise
  529. // just believe the api.
  530. //
  531. //
  532. DriveName[0] = Drive;
  533. rc = GetDriveType(DriveName);
  534. if((rc != DRIVE_REMOVABLE) || (Drive < L'C')) {
  535. return(rc);
  536. }
  537. //
  538. // DRIVE_REMOVABLE on NT.
  539. //
  540. //
  541. // Disallow use of removable media (e.g. Jazz, Zip, ...).
  542. //
  543. DriveNameNt[4] = Drive;
  544. hDisk = CreateFile(
  545. DriveNameNt,
  546. FILE_READ_ATTRIBUTES | SYNCHRONIZE,
  547. FILE_SHARE_READ | FILE_SHARE_WRITE,
  548. NULL,
  549. OPEN_EXISTING,
  550. 0,
  551. NULL
  552. );
  553. if(hDisk != INVALID_HANDLE_VALUE) {
  554. b = DeviceIoControl(
  555. hDisk,
  556. IOCTL_DISK_GET_DRIVE_GEOMETRY,
  557. NULL,
  558. 0,
  559. &MediaInfo,
  560. sizeof(MediaInfo),
  561. &DataSize,
  562. NULL
  563. );
  564. //
  565. // It's really a hard disk if the media type is removable.
  566. //
  567. if(b && (MediaInfo.MediaType == RemovableMedia)) {
  568. rc = DRIVE_FIXED;
  569. }
  570. CloseHandle(hDisk);
  571. }
  572. return(rc);
  573. }
  574. BOOL
  575. IsNEC98(
  576. VOID
  577. )
  578. {
  579. static BOOL Checked = FALSE;
  580. static BOOL Is98;
  581. if(!Checked) {
  582. Is98 = ((GetKeyboardType(0) == 7) && ((GetKeyboardType(1) & 0xff00) == 0x0d00));
  583. Checked = TRUE;
  584. }
  585. return(Is98);
  586. }