]> glassweightruler.freedombox.rocks Git - Ventoy.git/blob - Ventoy2Disk/Ventoy2Disk/Utility.c
ff77db5564313299802d925a59dc3a706fc72442
[Ventoy.git] / Ventoy2Disk / Ventoy2Disk / Utility.c
1 /******************************************************************************
2 * Utility.c
3 *
4 * Copyright (c) 2020, longpanda <admin@ventoy.net>
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License as
8 * published by the Free Software Foundation; either version 3 of the
9 * License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, see <http://www.gnu.org/licenses/>.
18 *
19 */
20 #include <Windows.h>
21 #include "Ventoy2Disk.h"
22
23 void Log(const char *Fmt, ...)
24 {
25 va_list Arg;
26 int Len = 0;
27 FILE *File = NULL;
28 SYSTEMTIME Sys;
29 char szBuf[1024];
30
31 GetLocalTime(&Sys);
32 Len += safe_sprintf(szBuf,
33 "[%4d/%02d/%02d %02d:%02d:%02d.%03d] ",
34 Sys.wYear, Sys.wMonth, Sys.wDay,
35 Sys.wHour, Sys.wMinute, Sys.wSecond,
36 Sys.wMilliseconds);
37
38 va_start(Arg, Fmt);
39 Len += vsnprintf_s(szBuf + Len, sizeof(szBuf)-Len, sizeof(szBuf)-Len, Fmt, Arg);
40 va_end(Arg);
41
42 //printf("%s\n", szBuf);
43
44 #if 1
45 fopen_s(&File, VENTOY_FILE_LOG, "a+");
46 if (File)
47 {
48 fwrite(szBuf, 1, Len, File);
49 fwrite("\n", 1, 1, File);
50 fclose(File);
51 }
52 #endif
53
54 }
55
56 BOOL IsPathExist(BOOL Dir, const char *Fmt, ...)
57 {
58 va_list Arg;
59 HANDLE hFile;
60 DWORD Attr;
61 CHAR FilePath[MAX_PATH];
62
63 va_start(Arg, Fmt);
64 vsnprintf_s(FilePath, sizeof(FilePath), sizeof(FilePath), Fmt, Arg);
65 va_end(Arg);
66
67 hFile = CreateFileA(FilePath, FILE_READ_EA, FILE_SHARE_READ, 0, OPEN_EXISTING, 0, 0);
68 if (INVALID_HANDLE_VALUE == hFile)
69 {
70 return FALSE;
71 }
72
73 CloseHandle(hFile);
74
75 Attr = GetFileAttributesA(FilePath);
76
77 if (Dir)
78 {
79 if ((Attr & FILE_ATTRIBUTE_DIRECTORY) == 0)
80 {
81 return FALSE;
82 }
83 }
84 else
85 {
86 if (Attr & FILE_ATTRIBUTE_DIRECTORY)
87 {
88 return FALSE;
89 }
90 }
91
92 return TRUE;
93 }
94
95
96 int ReadWholeFileToBuf(const CHAR *FileName, int ExtLen, void **Bufer, int *BufLen)
97 {
98 int FileSize;
99 FILE *File = NULL;
100 void *Data = NULL;
101
102 fopen_s(&File, FileName, "rb");
103 if (File == NULL)
104 {
105 Log("Failed to open file %s", FileName);
106 return 1;
107 }
108
109 fseek(File, 0, SEEK_END);
110 FileSize = (int)ftell(File);
111
112 Data = malloc(FileSize + ExtLen);
113 if (!Data)
114 {
115 fclose(File);
116 return 1;
117 }
118
119 fseek(File, 0, SEEK_SET);
120 fread(Data, 1, FileSize, File);
121
122 fclose(File);
123
124 *Bufer = Data;
125 *BufLen = FileSize;
126
127 return 0;
128 }
129
130 const CHAR* GetLocalVentoyVersion(void)
131 {
132 int rc;
133 int FileSize;
134 CHAR *Pos = NULL;
135 CHAR *Buf = NULL;
136 static CHAR LocalVersion[64] = { 0 };
137
138 if (LocalVersion[0] == 0)
139 {
140 rc = ReadWholeFileToBuf(VENTOY_FILE_VERSION, 1, (void **)&Buf, &FileSize);
141 if (rc)
142 {
143 return "";
144 }
145 Buf[FileSize] = 0;
146
147 for (Pos = Buf; *Pos; Pos++)
148 {
149 if (*Pos == '\r' || *Pos == '\n')
150 {
151 *Pos = 0;
152 break;
153 }
154 }
155
156 safe_sprintf(LocalVersion, "%s", Buf);
157 free(Buf);
158 }
159
160 return LocalVersion;
161 }
162
163 const CHAR* ParseVentoyVersionFromString(CHAR *Buf)
164 {
165 CHAR *Pos = NULL;
166 CHAR *End = NULL;
167 static CHAR LocalVersion[64] = { 0 };
168
169 Pos = strstr(Buf, "VENTOY_VERSION=");
170 if (Pos)
171 {
172 Pos += strlen("VENTOY_VERSION=");
173 if (*Pos == '"')
174 {
175 Pos++;
176 }
177
178 End = Pos;
179 while (*End != 0 && *End != '"' && *End != '\r' && *End != '\n')
180 {
181 End++;
182 }
183
184 *End = 0;
185
186 safe_sprintf(LocalVersion, "%s", Pos);
187 return LocalVersion;
188 }
189
190 return "";
191 }
192
193 BOOL IsWow64(void)
194 {
195 typedef BOOL(WINAPI *LPFN_ISWOW64PROCESS)(HANDLE, PBOOL);
196 LPFN_ISWOW64PROCESS fnIsWow64Process;
197 BOOL bIsWow64 = FALSE;
198
199 fnIsWow64Process = (LPFN_ISWOW64PROCESS)GetProcAddress(GetModuleHandleA("kernel32"), "IsWow64Process");
200 if (NULL != fnIsWow64Process)
201 {
202 fnIsWow64Process(GetCurrentProcess(), &bIsWow64);
203 }
204
205 return bIsWow64;
206 }
207
208 void DumpWindowsVersion(void)
209 {
210 int Bit;
211 BOOL WsVer;
212 DWORD Major, Minor;
213 ULONGLONG MajorEqual, MinorEqual;
214 OSVERSIONINFOEXA Ver1, Ver2;
215 const CHAR *Ver = NULL;
216 CHAR WinVer[256] = { 0 };
217
218 memset(&Ver1, 0, sizeof(Ver1));
219 memset(&Ver2, 0, sizeof(Ver2));
220
221 Ver1.dwOSVersionInfoSize = sizeof(Ver1);
222
223 // suppress the C4996 warning for GetVersionExA
224 #pragma warning(push)
225 #pragma warning(disable:4996)
226 if (!GetVersionExA((OSVERSIONINFOA *)&Ver1))
227 {
228 memset(&Ver1, 0, sizeof(Ver1));
229 Ver1.dwOSVersionInfoSize = sizeof(OSVERSIONINFOA);
230 if (!GetVersionExA((OSVERSIONINFOA *)&Ver1))
231 {
232 return;
233 }
234 }
235 #pragma warning(pop)
236
237 if (Ver1.dwPlatformId == VER_PLATFORM_WIN32_NT)
238 {
239 if (Ver1.dwMajorVersion > 6 || (Ver1.dwMajorVersion == 6 && Ver1.dwMinorVersion >= 2))
240 {
241 // GetVersionEx() has problem on some Windows version
242
243 MajorEqual = VerSetConditionMask(0, VER_MAJORVERSION, VER_EQUAL);
244 for (Major = Ver1.dwMajorVersion; Major <= 9; Major++)
245 {
246 memset(&Ver2, 0, sizeof(Ver2));
247 Ver2.dwOSVersionInfoSize = sizeof(Ver2);
248 Ver2.dwMajorVersion = Major;
249
250 if (!VerifyVersionInfoA(&Ver2, VER_MAJORVERSION, MajorEqual))
251 {
252 continue;
253 }
254
255 if (Ver1.dwMajorVersion < Major)
256 {
257 Ver1.dwMajorVersion = Major;
258 Ver1.dwMinorVersion = 0;
259 }
260
261 MinorEqual = VerSetConditionMask(0, VER_MINORVERSION, VER_EQUAL);
262 for (Minor = Ver1.dwMinorVersion; Minor <= 9; Minor++)
263 {
264 memset(&Ver2, 0, sizeof(Ver2));
265
266 Ver2.dwOSVersionInfoSize = sizeof(Ver2);
267 Ver2.dwMinorVersion = Minor;
268
269 if (!VerifyVersionInfoA(&Ver2, VER_MINORVERSION, MinorEqual))
270 {
271 continue;
272 }
273
274 Ver1.dwMinorVersion = Minor;
275 break;
276 }
277
278 break;
279 }
280 }
281
282 if (Ver1.dwMajorVersion <= 0xF && Ver1.dwMinorVersion <= 0xF)
283 {
284 WsVer = (Ver1.wProductType <= VER_NT_WORKSTATION);
285 switch ((Ver1.dwMajorVersion << 4) | Ver2.dwMinorVersion)
286 {
287 case 0x51:
288 {
289 Ver = "XP";
290 break;
291 }
292 case 0x52:
293 {
294 Ver = GetSystemMetrics(89) ? "Server 2003 R2" : "Server 2003";
295 break;
296 }
297 case 0x60:
298 {
299 Ver = WsVer ? "Vista" : "Server 2008";
300 break;
301 }
302 case 0x61:
303 {
304 Ver = WsVer ? "7" : "Server 2008 R2";
305 break;
306 }
307 case 0x62:
308 {
309 Ver = WsVer ? "8" : "Server 2012";
310 break;
311 }
312 case 0x63:
313 {
314 Ver = WsVer ? "8.1" : "Server 2012 R2";
315 break;
316 }
317 case 0x64:
318 {
319 Ver = WsVer ? "10 (Preview 1)" : "Server 10 (Preview 1)";
320 break;
321 }
322 case 0xA0:
323 {
324 Ver = WsVer ? "10" : ((Ver1.dwBuildNumber > 15000) ? "Server 2019" : "Server 2016");
325 break;
326 }
327 default:
328 {
329 Ver = "10 or later";
330 break;
331 }
332 }
333 }
334 }
335
336 Bit = IsWow64() ? 64 : 32;
337
338 if (Ver1.wServicePackMinor)
339 {
340 safe_sprintf(WinVer, "Windows %s SP%u.%u %d-bit", Ver, Ver1.wServicePackMajor, Ver1.wServicePackMinor, Bit);
341 }
342 else if (Ver1.wServicePackMajor)
343 {
344 safe_sprintf(WinVer, "Windows %s SP%u %d-bit", Ver, Ver1.wServicePackMajor, Bit);
345 }
346 else
347 {
348 safe_sprintf(WinVer, "Windows %s %d-bit", Ver, Bit);
349 }
350
351 if (((Ver1.dwMajorVersion << 4) | Ver2.dwMinorVersion) >= 0x62)
352 {
353 Log("Windows Version : %s (Build %u)", WinVer, Ver1.dwBuildNumber);
354 }
355 else
356 {
357 Log("Windows Version : %s", WinVer);
358 }
359
360 return;
361 }
362
363 BOOL IsVentoyLogicalDrive(CHAR DriveLetter)
364 {
365 int i;
366 CONST CHAR *Files[] =
367 {
368 "EFI\\BOOT\\BOOTX64.EFI",
369 "grub\\themes\\ventoy\\theme.txt",
370 "ventoy\\ventoy.cpio",
371 };
372
373 for (i = 0; i < sizeof(Files) / sizeof(Files[0]); i++)
374 {
375 if (!IsFileExist("%C:\\%s", DriveLetter, Files[i]))
376 {
377 return FALSE;
378 }
379 }
380
381 return TRUE;
382 }
383
384
385 int VentoyFillLocation(UINT64 DiskSizeInBytes, UINT32 StartSectorId, UINT32 SectorCount, PART_TABLE *Table)
386 {
387 BYTE Head;
388 BYTE Sector;
389 BYTE nSector = 63;
390 BYTE nHead = 8;
391 UINT32 Cylinder;
392 UINT32 EndSectorId;
393
394 while (nHead != 0 && (DiskSizeInBytes / 512 / nSector / nHead) > 1024)
395 {
396 nHead = (BYTE)nHead * 2;
397 }
398
399 if (nHead == 0)
400 {
401 nHead = 255;
402 }
403
404 Cylinder = StartSectorId / nSector / nHead;
405 Head = StartSectorId / nSector % nHead;
406 Sector = StartSectorId % nSector + 1;
407
408 Table->StartHead = Head;
409 Table->StartSector = Sector;
410 Table->StartCylinder = Cylinder;
411
412 EndSectorId = StartSectorId + SectorCount - 1;
413 Cylinder = EndSectorId / nSector / nHead;
414 Head = EndSectorId / nSector % nHead;
415 Sector = EndSectorId % nSector + 1;
416
417 Table->EndHead = Head;
418 Table->EndSector = Sector;
419 Table->EndCylinder = Cylinder;
420
421 Table->StartSectorId = StartSectorId;
422 Table->SectorCount = SectorCount;
423
424 return 0;
425 }
426
427 int VentoyFillMBR(UINT64 DiskSizeBytes, MBR_HEAD *pMBR, int PartStyle)
428 {
429 GUID Guid;
430 int ReservedValue;
431 UINT32 DiskSignature;
432 UINT32 DiskSectorCount;
433 UINT32 PartSectorCount;
434 UINT32 PartStartSector;
435 UINT32 ReservedSector;
436
437 VentoyGetLocalBootImg(pMBR);
438
439 CoCreateGuid(&Guid);
440
441 memcpy(&DiskSignature, &Guid, sizeof(UINT32));
442
443 Log("Disk signature: 0x%08x", DiskSignature);
444
445 *((UINT32 *)(pMBR->BootCode + 0x1B8)) = DiskSignature;
446
447 if (DiskSizeBytes / 512 > 0xFFFFFFFF)
448 {
449 DiskSectorCount = 0xFFFFFFFF;
450 }
451 else
452 {
453 DiskSectorCount = (UINT32)(DiskSizeBytes / 512);
454 }
455
456 ReservedValue = GetReservedSpaceInMB();
457 if (ReservedValue <= 0)
458 {
459 ReservedSector = 0;
460 }
461 else
462 {
463 ReservedSector = (UINT32)(ReservedValue * 2048);
464 }
465
466 if (PartStyle)
467 {
468 ReservedSector += 33; // backup GPT part table
469 }
470
471 Log("ReservedSector: %u", ReservedSector);
472
473 //Part1
474 PartStartSector = VENTOY_PART1_START_SECTOR;
475 PartSectorCount = DiskSectorCount - ReservedSector - VENTOY_EFI_PART_SIZE / 512 - PartStartSector;
476 VentoyFillLocation(DiskSizeBytes, PartStartSector, PartSectorCount, pMBR->PartTbl);
477
478 pMBR->PartTbl[0].Active = 0x80; // bootable
479 pMBR->PartTbl[0].FsFlag = 0x07; // exFAT/NTFS/HPFS
480
481 //Part2
482 PartStartSector += PartSectorCount;
483 PartSectorCount = VENTOY_EFI_PART_SIZE / 512;
484 VentoyFillLocation(DiskSizeBytes, PartStartSector, PartSectorCount, pMBR->PartTbl + 1);
485
486 pMBR->PartTbl[1].Active = 0x00;
487 pMBR->PartTbl[1].FsFlag = 0xEF; // EFI System Partition
488
489 pMBR->Byte55 = 0x55;
490 pMBR->ByteAA = 0xAA;
491
492 return 0;
493 }
494
495
496 static int VentoyFillProtectMBR(UINT64 DiskSizeBytes, MBR_HEAD *pMBR)
497 {
498 GUID Guid;
499 UINT32 DiskSignature;
500 UINT64 DiskSectorCount;
501
502 VentoyGetLocalBootImg(pMBR);
503
504 CoCreateGuid(&Guid);
505
506 memcpy(&DiskSignature, &Guid, sizeof(UINT32));
507
508 Log("Disk signature: 0x%08x", DiskSignature);
509
510 *((UINT32 *)(pMBR->BootCode + 0x1B8)) = DiskSignature;
511
512 DiskSectorCount = DiskSizeBytes / 512 - 1;
513 if (DiskSectorCount > 0xFFFFFFFF)
514 {
515 DiskSectorCount = 0xFFFFFFFF;
516 }
517
518 memset(pMBR->PartTbl, 0, sizeof(pMBR->PartTbl));
519
520 pMBR->PartTbl[0].Active = 0x00;
521 pMBR->PartTbl[0].FsFlag = 0xee; // EE
522
523 pMBR->PartTbl[0].StartHead = 0;
524 pMBR->PartTbl[0].StartSector = 1;
525 pMBR->PartTbl[0].StartCylinder = 0;
526 pMBR->PartTbl[0].EndHead = 254;
527 pMBR->PartTbl[0].EndSector = 63;
528 pMBR->PartTbl[0].EndCylinder = 1023;
529
530 pMBR->PartTbl[0].StartSectorId = 1;
531 pMBR->PartTbl[0].SectorCount = (UINT32)DiskSectorCount;
532
533 pMBR->Byte55 = 0x55;
534 pMBR->ByteAA = 0xAA;
535
536 pMBR->BootCode[92] = 0x22;
537
538 return 0;
539 }
540
541
542 int VentoyFillGpt(UINT64 DiskSizeBytes, VTOY_GPT_INFO *pInfo)
543 {
544 INT64 ReservedValue = 0;
545 UINT64 ReservedSector = 33;
546 UINT64 Part1SectorCount = 0;
547 UINT64 DiskSectorCount = DiskSizeBytes / 512;
548 VTOY_GPT_HDR *Head = &pInfo->Head;
549 VTOY_GPT_PART_TBL *Table = pInfo->PartTbl;
550 static GUID WindowsDataPartType = { 0xebd0a0a2, 0xb9e5, 0x4433, { 0x87, 0xc0, 0x68, 0xb6, 0xb7, 0x26, 0x99, 0xc7 } };
551 static GUID EspPartType = { 0xc12a7328, 0xf81f, 0x11d2, { 0xba, 0x4b, 0x00, 0xa0, 0xc9, 0x3e, 0xc9, 0x3b } };
552 static GUID BiosGrubPartType = { 0x21686148, 0x6449, 0x6e6f, { 0x74, 0x4e, 0x65, 0x65, 0x64, 0x45, 0x46, 0x49 } };
553
554 VentoyFillProtectMBR(DiskSizeBytes, &pInfo->MBR);
555
556 ReservedValue = GetReservedSpaceInMB();
557 if (ReservedValue > 0)
558 {
559 ReservedSector += ReservedValue * 2048;
560 }
561
562 Part1SectorCount = DiskSectorCount - ReservedSector - (VENTOY_EFI_PART_SIZE / 512) - 2048;
563
564 memcpy(Head->Signature, "EFI PART", 8);
565 Head->Version[2] = 0x01;
566 Head->Length = 92;
567 Head->Crc = 0;
568 Head->EfiStartLBA = 1;
569 Head->EfiBackupLBA = DiskSectorCount - 1;
570 Head->PartAreaStartLBA = 34;
571 Head->PartAreaEndLBA = DiskSectorCount - 34;
572 CoCreateGuid(&Head->DiskGuid);
573 Head->PartTblStartLBA = 2;
574 Head->PartTblTotNum = 128;
575 Head->PartTblEntryLen = 128;
576
577
578 memcpy(&(Table[0].PartType), &WindowsDataPartType, sizeof(GUID));
579 CoCreateGuid(&(Table[0].PartGuid));
580 Table[0].StartLBA = 2048;
581 Table[0].LastLBA = 2048 + Part1SectorCount - 1;
582 Table[0].Attr = 0;
583 memcpy(Table[0].Name, L"Ventoy", 6 * 2);
584
585 // to fix windows issue
586 //memcpy(&(Table[1].PartType), &EspPartType, sizeof(GUID));
587 memcpy(&(Table[1].PartType), &WindowsDataPartType, sizeof(GUID));
588 CoCreateGuid(&(Table[1].PartGuid));
589 Table[1].StartLBA = Table[0].LastLBA + 1;
590 Table[1].LastLBA = Table[1].StartLBA + VENTOY_EFI_PART_SIZE / 512 - 1;
591 Table[1].Attr = 0x8000000000000001ULL;
592 memcpy(Table[1].Name, L"VTOYEFI", 7 * 2);
593
594 #if 0
595 memcpy(&(Table[2].PartType), &BiosGrubPartType, sizeof(GUID));
596 CoCreateGuid(&(Table[2].PartGuid));
597 Table[2].StartLBA = 34;
598 Table[2].LastLBA = 2047;
599 Table[2].Attr = 0;
600 #endif
601
602 //Update CRC
603 Head->PartTblCrc = VentoyCrc32(Table, sizeof(pInfo->PartTbl));
604 Head->Crc = VentoyCrc32(Head, Head->Length);
605
606 return 0;
607 }
608
609 int VentoyFillBackupGptHead(VTOY_GPT_INFO *pInfo, VTOY_GPT_HDR *pHead)
610 {
611 UINT64 LBA;
612 UINT64 BackupLBA;
613
614 memcpy(pHead, &pInfo->Head, sizeof(VTOY_GPT_HDR));
615
616 LBA = pHead->EfiStartLBA;
617 BackupLBA = pHead->EfiBackupLBA;
618
619 pHead->EfiStartLBA = BackupLBA;
620 pHead->EfiBackupLBA = LBA;
621 pHead->PartTblStartLBA = BackupLBA + 1 - 33;
622
623 pHead->Crc = 0;
624 pHead->Crc = VentoyCrc32(pHead, pHead->Length);
625
626 return 0;
627 }
628
629 CHAR GetFirstUnusedDriveLetter(void)
630 {
631 CHAR Letter = 'D';
632 DWORD Drives = GetLogicalDrives();
633
634 Drives >>= 3;
635 while (Drives & 0x1)
636 {
637 Letter++;
638 Drives >>= 1;
639 }
640
641 return Letter;
642 }
643
644 const CHAR * GetBusTypeString(STORAGE_BUS_TYPE Type)
645 {
646 switch (Type)
647 {
648 case BusTypeUnknown: return "unknown";
649 case BusTypeScsi: return "SCSI";
650 case BusTypeAtapi: return "Atapi";
651 case BusTypeAta: return "ATA";
652 case BusType1394: return "1394";
653 case BusTypeSsa: return "SSA";
654 case BusTypeFibre: return "Fibre";
655 case BusTypeUsb: return "USB";
656 case BusTypeRAID: return "RAID";
657 case BusTypeiScsi: return "iSCSI";
658 case BusTypeSas: return "SAS";
659 case BusTypeSata: return "SATA";
660 case BusTypeSd: return "SD";
661 case BusTypeMmc: return "MMC";
662 case BusTypeVirtual: return "Virtual";
663 case BusTypeFileBackedVirtual: return "FileBackedVirtual";
664 case BusTypeSpaces: return "Spaces";
665 case BusTypeNvme: return "Nvme";
666 }
667 return "unknown";
668 }
669
670 int VentoyGetLocalBootImg(MBR_HEAD *pMBR)
671 {
672 int Len = 0;
673 BYTE *ImgBuf = NULL;
674 static int Loaded = 0;
675 static MBR_HEAD MBR;
676
677 if (Loaded)
678 {
679 memcpy(pMBR, &MBR, 512);
680 return 0;
681 }
682
683 if (0 == ReadWholeFileToBuf(VENTOY_FILE_BOOT_IMG, 0, (void **)&ImgBuf, &Len))
684 {
685 Log("Copy boot img success");
686 memcpy(pMBR, ImgBuf, 512);
687 free(ImgBuf);
688
689 CoCreateGuid((GUID *)(pMBR->BootCode + 0x180));
690
691 memcpy(&MBR, pMBR, 512);
692 Loaded = 1;
693
694 return 0;
695 }
696 else
697 {
698 Log("Copy boot img failed");
699 return 1;
700 }
701 }
702
703 int GetHumanReadableGBSize(UINT64 SizeBytes)
704 {
705 int i;
706 int Pow2 = 1;
707 double Delta;
708 double GB = SizeBytes * 1.0 / 1000 / 1000 / 1000;
709
710 for (i = 0; i < 12; i++)
711 {
712 if (Pow2 > GB)
713 {
714 Delta = (Pow2 - GB) / Pow2;
715 }
716 else
717 {
718 Delta = (GB - Pow2) / Pow2;
719 }
720
721 if (Delta < 0.05)
722 {
723 return Pow2;
724 }
725
726 Pow2 <<= 1;
727 }
728
729 return (int)GB;
730 }
731
732 void TrimString(CHAR *String)
733 {
734 CHAR *Pos1 = String;
735 CHAR *Pos2 = String;
736 size_t Len = strlen(String);
737
738 while (Len > 0)
739 {
740 if (String[Len - 1] != ' ' && String[Len - 1] != '\t')
741 {
742 break;
743 }
744 String[Len - 1] = 0;
745 Len--;
746 }
747
748 while (*Pos1 == ' ' || *Pos1 == '\t')
749 {
750 Pos1++;
751 }
752
753 while (*Pos1)
754 {
755 *Pos2++ = *Pos1++;
756 }
757 *Pos2++ = 0;
758
759 return;
760 }
761
762 int GetRegDwordValue(HKEY Key, LPCSTR SubKey, LPCSTR ValueName, DWORD *pValue)
763 {
764 HKEY hKey;
765 DWORD Type;
766 DWORD Size;
767 LSTATUS lRet;
768 DWORD Value;
769
770 lRet = RegOpenKeyExA(Key, SubKey, 0, KEY_QUERY_VALUE, &hKey);
771 Log("RegOpenKeyExA <%s> Ret:%ld", SubKey, lRet);
772
773 if (ERROR_SUCCESS == lRet)
774 {
775 Size = sizeof(Value);
776 lRet = RegQueryValueExA(hKey, ValueName, NULL, &Type, (LPBYTE)&Value, &Size);
777 Log("RegQueryValueExA <%s> ret:%u Size:%u Value:%u", ValueName, lRet, Size, Value);
778
779 *pValue = Value;
780 RegCloseKey(hKey);
781
782 return 0;
783 }
784 else
785 {
786 return 1;
787 }
788 }
789
790 int GetPhysicalDriveCount(void)
791 {
792 DWORD Value;
793 int Count = 0;
794
795 if (GetRegDwordValue(HKEY_LOCAL_MACHINE, "SYSTEM\\CurrentControlSet\\Services\\disk\\Enum", "Count", &Value) == 0)
796 {
797 Count = (int)Value;
798 }
799
800 Log("GetPhysicalDriveCount: %d", Count);
801 return Count;
802 }
803
804
805