Попытка написать всеобъемлющий метод для определения типа диска Windows, используя C++
Я встречал тонну неуправляемых C++/MFC примеров людей, пытающихся определить тип диска по его пути DOS, но, похоже, существует много путаницы по поводу того, как это должно быть сделано. Поэтому я пытаюсь объединить всю информацию в одну функцию и сделать то, что Microsoft должна была сделать для нас давным-давно.
Мне в основном нужно отнести все современные диски к этим:
- Фиксированный диск
- Съемный диск
- Медиа диск
- Другое (см. Объяснение в коде)
Может кто-нибудь проверить мой код ниже?
enum DRIVE_TYPE2{
DT2_ERROR,
DT2_FIXED_DRIVE, //Any internal drive that cannot be removed while system is running, like a hard drive
DT2_REMOVABLE_DRIVE, //Removable drive. Example: Any USB drive
DT2_MEDIA_DRIVE, //CD/DVD drive, or a floppy drive
DT2_OTHER, //Other (rare) types, like virtual drive, RAM drive, etc.
};
static DRIVE_TYPE2 getDriveType(LPCTSTR pStrDriveDosPath)
{
//Determine the drive type by its DOS path
//'pStrDriveDosPath' = Path to the drive (example: "c:" or "d:") -- must be only a local drive letter!
//RETURN:
// = One of the values from the DRIVE_TYPE2 enum
ASSERT(pStrDriveDosPath);
DRIVE_TYPE2 res = DT2_ERROR;
//Make path
CString strPath;
strPath.Format(_T("\\\\.\\%s"), pStrDriveDosPath);
HANDLE hDevice = CreateFile(strPath, 0, FILE_SHARE_READ | FILE_SHARE_WRITE,
NULL, OPEN_EXISTING, 0, NULL);
if ( hDevice != INVALID_HANDLE_VALUE )
{
//First check for floppy discs
DISK_GEOMETRY Geom[20];
DWORD cb;
if(DeviceIoControl(hDevice, IOCTL_DISK_GET_MEDIA_TYPES, 0, 0, Geom, sizeof(Geom), &cb, 0)
&& cb > 0)
{
switch (Geom[0].MediaType)
{
case F5_1Pt2_512: // 5.25 1.2MB floppy
case F5_360_512: // 5.25 360K floppy
case F5_320_512: // 5.25 320K floppy
case F5_320_1024: // 5.25 320K floppy
case F5_180_512: // 5.25 180K floppy
case F5_160_512: // 5.25 160K floppy
case F5_640_512: //5.25", 640 KB, 512 bytes/sector media
case F5_720_512: //5.25", 720 KB, 512 bytes/sector media
case F5_1Pt23_1024: //5.25", 1.23 MB, 1024 bytes/sector media
case F3_1Pt44_512: // 3.5 1.44MB floppy
case F3_2Pt88_512: // 3.5 2.88MB floppy
case F3_20Pt8_512: // 3.5 20.8MB floppy
case F3_720_512: // 3.5 720K floppy
case F3_120M_512: //3.5", 120 MB media
case F3_640_512: //3.5", 640 KB, 512 bytes/sector media
case F3_1Pt2_512: //3.5", 1.2 MB, 512 bytes/sector media
case F3_1Pt23_1024: //3.5", 1.23 MB, 1024 bytes/sector media
case F3_128Mb_512: //3.5" magneto-optical (MO), 128 MB, 512 bytes/sector media
case F3_230Mb_512: //3.5" magneto-optical (MO), 230 MB, 512 bytes/sector media
case F8_256_128: //8", 256 KB, 128 bytes/sector media
case RemovableMedia:
res = DT2_MEDIA_DRIVE;
break;
}
}
//Did we get the type?
if(res == DT2_ERROR)
{
//Check for CD/DVD drive
UINT nDrvType = GetDriveType(pStrDriveDosPath + CString(_T("\\")));
switch(nDrvType)
{
case DRIVE_CDROM:
//Media drive
res = DT2_MEDIA_DRIVE;
break;
}
//Did we get the type?
if(res == DT2_ERROR)
{
//Then try by bus type
DWORD dwOutBytes = 0; // IOCTL output length
STORAGE_PROPERTY_QUERY Query; // input param for query
// specify the query type
Query.PropertyId = StorageDeviceProperty;
Query.QueryType = PropertyStandardQuery;
char OutBuf[1024] = {0}; // good enough, usually about 100 bytes
PSTORAGE_DEVICE_DESCRIPTOR pDevDesc = (PSTORAGE_DEVICE_DESCRIPTOR)OutBuf;
pDevDesc->Size = sizeof(OutBuf);
// Query using IOCTL_STORAGE_QUERY_PROPERTY
if(DeviceIoControl(hDevice, // device handle
IOCTL_STORAGE_QUERY_PROPERTY, // info of device property
&Query, sizeof(STORAGE_PROPERTY_QUERY), // input data buffer
pDevDesc, pDevDesc->Size, // output data buffer
&dwOutBytes, // out's length
(LPOVERLAPPED)NULL))
{
//See by type
switch(pDevDesc->BusType)
{
case BusTypeAtapi:
case BusTypeAta:
case BusTypeRAID:
case BusTypeSas:
case BusTypeSata:
//Fixed
res = DT2_FIXED_DRIVE;
break;
case BusTypeScsi:
case BusType1394: //Firewire
case BusTypeSsa:
case BusTypeFibre:
case BusTypeUsb:
case BusTypeiScsi:
case BusTypeSd:
case BusTypeMmc:
//Removable
res = DT2_REMOVABLE_DRIVE;
break;
case BusTypeVirtual:
case BusTypeFileBackedVirtual:
//Other
res = DT2_OTHER;
break;
default:
//Error
ASSERT(pDevDesc->BusType == BusTypeUnknown);
res = DT2_ERROR;
break;
}
}
}
}
//Close drive handle
CloseHandle(hDevice);
}
return res;
}
1 ответ
Microsoft уже написала именно эту функцию (хотя они не разделяли категории точно так же)
GetDriveType