磁盘设备类型获取函数

//将该部分内容保存到.cpp文件中可直接编译运行,用于辨别驱动器的类型!

 

#define MEDIA_INFO_SIZE    sizeof(GET_MEDIA_TYPES)+15*sizeof(DEVICE_MEDIA_INFO)
#define IOCTL_STORAGE_QUERY_PROPERTY   CTL_CODE(IOCTL_STORAGE_BASE, 0x0500, METHOD_BUFFERED, FILE_ANY_ACCESS)

typedef enum _STORAGE_BUS_TYPE { 
 BusTypeUnknown = 0x00, 
  BusTypeScsi, 
  BusTypeAtapi, 
  BusTypeAta, 
  BusType1394, 
  BusTypeSsa, 
  BusTypeFibre, 
  BusTypeUsb, 
  BusTypeRAID, 
  BusTypeiScsi, 
  BusTypeSas, 
  BusTypeSata, 
  BusTypeSd, 
  BusTypeMmc, 
  BusTypeMax, 
  BusTypeMaxReserved = 0x7F
} STORAGE_BUS_TYPE, *PSTORAGE_BUS_TYPE;


typedef enum _STORAGE_PROPERTY_ID {
 StorageDeviceProperty = 0,
  StorageAdapterProperty,
  StorageDeviceIdProperty
} STORAGE_PROPERTY_ID, *PSTORAGE_PROPERTY_ID;

typedef enum _STORAGE_QUERY_TYPE {
 PropertyStandardQuery = 0,
  PropertyExistsQuery,
  PropertyMaskQuery,
  PropertyQueryMaxDefined
} STORAGE_QUERY_TYPE, *PSTORAGE_QUERY_TYPE;


typedef struct _STORAGE_PROPERTY_QUERY {
 STORAGE_PROPERTY_ID  PropertyId;
 STORAGE_QUERY_TYPE  QueryType;
 unsigned char  AdditionalParameters[1];
} STORAGE_PROPERTY_QUERY, *PSTORAGE_PROPERTY_QUERY;


typedef struct _STORAGE_DEVICE_DESCRIPTOR {
 unsigned long  Version;
 unsigned long  Size;
 unsigned char DeviceType;
 unsigned char DeviceTypeModifier;
 unsigned char  RemovableMedia;
 unsigned char  CommandQueueing;
 unsigned long  VendorIdOffset;
 unsigned long  ProductIdOffset;
 unsigned long  ProductRevisionOffset;
 unsigned long  SerialNumberOffset;
 STORAGE_BUS_TYPE  BusType;
 unsigned long  RawPropertiesLength;
 unsigned char RawDeviceProperties[1];
} STORAGE_DEVICE_DESCRIPTOR, *PSTORAGE_DEVICE_DESCRIPTOR;

#include <windows.h>
#include <winioctl.h>
#include <iostream.h>
#include <string.h>
#include <tchar.h>

 

BOOL GetDriveGeometry(const TCHAR * filename, DISK_GEOMETRY * pdg)
{
 HANDLE hDevice;         // 设备句柄
 BOOL bResult;           // DeviceIoControl的返回结果
 GET_MEDIA_TYPES *pmt;   // 内部用的输出缓冲区
 DWORD dwOutBytes;       // 输出数据长度
 
 // 打开设备
 hDevice = ::CreateFile(filename,           // 文件名
  GENERIC_READ,                          // 软驱需要读盘
  FILE_SHARE_READ | FILE_SHARE_WRITE,    // 共享方式
  NULL,                                  // 默认的安全描述符
  OPEN_EXISTING,                         // 创建方式
  0,                                     // 不需设置文件属性
  NULL);                                 // 不需参照模板文件
 
 if (hDevice == INVALID_HANDLE_VALUE)
 {
  // 设备无法打开...
  return FALSE;
 }
 
 // 用IOCTL_DISK_GET_DRIVE_GEOMETRY取磁盘参数
 bResult = ::DeviceIoControl(hDevice,       // 设备句柄
  IOCTL_DISK_GET_DRIVE_GEOMETRY,         // 取磁盘参数
  NULL, 0,                               // 不需要输入数据
  pdg, sizeof(DISK_GEOMETRY),            // 输出数据缓冲区
  &dwOutBytes,                           // 输出数据长度
  (LPOVERLAPPED)NULL);                   // 用同步I/O
 
 // 如果失败,再用IOCTL_STORAGE_GET_MEDIA_TYPES_EX取介质类型参数
 if (!bResult)
 {
  pmt = (GET_MEDIA_TYPES *)new BYTE[MEDIA_INFO_SIZE];
  
  bResult = ::DeviceIoControl(hDevice,    // 设备句柄
   IOCTL_STORAGE_GET_MEDIA_TYPES_EX,   // 取介质类型参数
   NULL, 0,                            // 不需要输入数据
   pmt, MEDIA_INFO_SIZE,               // 输出数据缓冲区
   &dwOutBytes,                        // 输出数据长度
   (LPOVERLAPPED)NULL);                // 用同步I/O
  
  if (bResult)
  {
   // 注意到结构DEVICE_MEDIA_INFO是在结构DISK_GEOMETRY的基础上扩充的
   // 为简化程序,用memcpy代替如下多条赋值语句:
   // pdg->MediaType = (MEDIA_TYPE)pmt->MediaInfo[0].DeviceSpecific.DiskInfo.MediaType;
   // pdg->Cylinders = pmt->MediaInfo[0].DeviceSpecific.DiskInfo.Cylinders;
   // pdg->TracksPerCylinder = pmt->MediaInfo[0].DeviceSpecific.DiskInfo.TracksPerCylinder;
   // ... ...
   ::memcpy(pdg, pmt->MediaInfo, sizeof(DISK_GEOMETRY));
  }
  
  delete pmt;
 }
 
 // 关闭设备句柄
 ::CloseHandle(hDevice);
 
 return (bResult);
 
}

 


int main()
{
 TCHAR szBuf[256];
 memset(szBuf,0,sizeof(szBuf));
 DISK_GEOMETRY dg;
 TCHAR sztmpPath[100] = _T(".//");
 TCHAR szPath[100] = _T(".//");
 HANDLE hDevice;         // 设备句柄
 BOOL bResult;           // DeviceIoControl的返回结果
 STORAGE_PROPERTY_QUERY Query; // input param for query
 DWORD dwOutBytes; // IOCTL output length
 STORAGE_DEVICE_DESCRIPTOR pDevDesc;

 

 DWORD len = GetLogicalDriveStrings(sizeof(szBuf)/sizeof(TCHAR),szBuf);
 
 for (TCHAR* s = szBuf; *s; s += strlen(s)+1)
 {
  LPCTSTR sDrivePath = s;
  memcpy(szPath,sztmpPath,sizeof(szPath));
  ::_tcscat(szPath,sDrivePath); 
  int nSize = ::_tcslen(szPath);
  szPath[nSize-1] = '/0';  
        BOOL bResult = GetDriveGeometry(szPath,&dg);
  
  // 打开设备
  hDevice = ::CreateFile(szPath,             // 文件名
   GENERIC_READ,                          // 软驱需要读盘
   FILE_SHARE_READ | FILE_SHARE_WRITE,    // 共享方式
   NULL,                                  // 默认的安全描述符
   OPEN_EXISTING,                         // 创建方式
   0,                                     // 不需设置文件属性
   NULL);   
  if (hDevice == INVALID_HANDLE_VALUE)
  {
   return FALSE;
  }
  
  
  Query.PropertyId = StorageDeviceProperty;
  Query.QueryType = PropertyStandardQuery;
    
  pDevDesc.Size = sizeof(STORAGE_DEVICE_DESCRIPTOR);

  
  // 用 IOCTL_STORAGE_QUERY_PROPERTY  
  bResult = ::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);

  if (bResult == FALSE)
  {
   CloseHandle(hDevice);
   return 0;
  }
  
  UINT Type = pDevDesc.BusType;
  switch(Type)
  {
  case    BusTypeUnknown:
   cout<<sDrivePath<<"  未知设备"<<'/n';
   break;

  case    BusTypeScsi:
   cout<<sDrivePath<<"  SCSI设备"<<'/n';
   break;
   
  case    BusTypeAtapi:
   cout<<sDrivePath<<"  API设备"<<'/n';
   break;
   
  case  BusTypeAta:
   cout<<sDrivePath<<"  ATA设备"<<'/n';
   break;
   
  case  BusType1394:
   cout<<sDrivePath<<"  1394设备"<<'/n';
   break;
   
  case BusTypeSsa:
   cout<<sDrivePath<<"  SSA设备"<<'/n';
   break;
   
  case  BusTypeFibre:
   cout<<sDrivePath<<"  光纤设备"<<'/n';
   break;
   
  case  BusTypeUsb:
   cout<<sDrivePath<<"  USB设备"<<'/n';
   break;
   
  case  BusTypeRAID:
   cout<<sDrivePath<<"  RAID设备"<<'/n';
   break;
   
  case BusTypeiScsi:
   cout<<sDrivePath<<"  iSCSI设备"<<'/n';
   break;
   
  case BusTypeSas:
   cout<<sDrivePath<<"  SAS设备"<<'/n';
   break;
   
  case BusTypeSata:
   cout<<sDrivePath<<"  SATA设备"<<'/n';
   break;
   
  case  BusTypeSd:
   cout<<sDrivePath<<"  SD设备"<<'/n';
   break;
   
  case BusTypeMmc:
   cout<<sDrivePath<<"  MMC设备"<<'/n';
   break;
   
  case BusTypeMax:
   cout<<sDrivePath<<"  MAX设备"<<'/n';
   break;
   
  case  BusTypeMaxReserved:
   cout<<sDrivePath<<"  ...设备"<<'/n';
   break;
  default:
   break;
  }
  CloseHandle(hDevice);
 }
};

 

欢迎大家加入"数据恢复技术"QQ群:30481379 

        本QQ群主要研究分析NTFS、EXT2/3、ResierFS等文件系统及数据恢复技术,同时对各种系统的磁盘阵列恢复的阵列参数智能分析算法进行技术交流,欢迎有志同道合者加入!

         由于本群主要面向数据恢复的技术研究,同时起点要高于一般的数据恢复技术,所以希望加入者在加入本群以前已经有关于这方面技术的积累会相关研究经验,希望大家能理解!

本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/ddl007/archive/2008/02/28/2127981.aspx

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
现在USB设备却很多,因此对USB设备的查找与读写就必不可少了。但是能找到关于USB读写的资料很少。这里使用VC++示范了一些获得USB的信息的方法。 一、枚举USB设备   通过枚举USB控制器->枚举此控制器上的USB HUB->枚举HUB的各个端口->获得设备信息。 枚举控制器: wsprintf(HCName, "\\\\.\\HCD%d", HCNum); hHCDev = CreateFile(HCName, GENERIC_WRITE, FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL);   计算机上的USB主控制器以HCD1,HCD2等命名。通过控制器名称,使用CCreateFile 打开它。使用DeviceIoControl即可得到其驱动程序名,以及与它连接的HUB的名称。用CCreateFile打开HUB,获得连接信息。再枚举HUB的各个端口即可获得连接的设备信息。 二、枚举HID设备   HID设备是微软定义的标准人机接口规范。比如USB鼠标,USB游戏手柄等。不用查找具体设备的GUID,使用API HidD_GetHidGuid(&guidHID)即可得到GUID。有了GUID通过API SetupDiEnumDeviceInterfaces可获得是否有设备连接。如果此类设备连接通过SetupDiGetDeviceInterfaceDetail获得它的设备路径信息。使用CCreateFile 打开它,通过HidD_GetAttributes获得其基本属性信息。使用DeviceIoControl可以获得更详细的属性。在本代码中如果计算机上插有USB游戏手柄,可获得其信息。但不知道为什么xp下却不能获得USB鼠标的信息。 三 枚举U盘   先用GetDriveType API获得设备类型,若类型为REMOVABLE(当然有些大容量U盘可能报告为FIXED,那就需要其他方法来确定了),即可能是U盘。用CCreateFile 打开之后,再用IOCTL_STORAGE_QUERY_PROPERTY为参数的DeviceIoControl来获得其属性。 四、结束语   示例工程在winxp+xpDDK+VC6下编译通过。USB设备种类比较多,也比较特殊,不同厂商的硬件不同,控制软件也不尽相同(我想主要是ICTL码不同,也不容易查到)。使得访问USB口的设备不象串口并口那么方便。这个例程只是展示了访问的基本方法。其中还有些问题还没有解决,发出来希望大家解决之后能通知我或者发表出来。 参考了USBPort,USBview等代码,一并致谢。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值