VC实现光驱、软驱、USB的禁用和启用

转载 2006年05月25日 11:30:00
由于工作的实际需要,需要对光驱,软驱,USB的启用和禁用实现控制。参考大家提供的对网卡禁用的代码,实现了光驱,软驱,USB的启用和禁用。主要实现代码如下:

// 必要的头文件和要链接的LIB文件
#include <setupapi.h>
#include <shlwapi.h>
#pragma comment(lib, "setupapi.lib")
#pragma comment(lib, "shlwapi.lib")
// device information set(我把它译为设备信息集)
HDEVINFO hDevInfo = NULL;

// 出错信息
void FormatMSG(DWORD dwError, LPTSTR * lpszMsg)
{
 BOOL bOk = FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ALLOCATE_BUFFER, NULL, dwError,
    MAKELANGID(LANG_CHINESE, SUBLANG_CHINESE_SIMPLIFIED), (LPTSTR)lpszMsg, 0, NULL);

 if (!bOk)
 {
  HMODULE hDll = LoadLibraryEx(_T("netmsg.dll"),
     NULL,
     DONT_RESOLVE_DLL_REFERENCES);
  if (NULL != hDll)
  {
    FormatMessage(FORMAT_MESSAGE_FROM_HMODULE |
        FORMAT_MESSAGE_FROM_SYSTEM,
        hDll,
        dwError,
        MAKELANGID(LANG_CHINESE, SUBLANG_CHINESE_SIMPLIFIED),
        (LPTSTR)lpszMsg,0, NULL);
    FreeLibrary(hDll);
  }
 }
}

BOOL ChangeStatus(DWORD NewStatus, DWORD SelectedItem, HDEVINFO hDevInfo)
{
 LPTSTR lpszMsg = NULL;
 HCURSOR hCursor = NULL;
 try
 {
  SP_PROPCHANGE_PARAMS PropChangeParams = {sizeof(SP_CLASSINSTALL_HEADER)};
  SP_DEVINFO_DATA DeviceInfoData = {sizeof(SP_DEVINFO_DATA)};

  hCursor = SetCursor(LoadCursor(NULL, IDC_WAIT));

  // Get a handle to the Selected Item.
  if (!SetupDiEnumDeviceInfo(hDevInfo, SelectedItem, &DeviceInfoData))
  {
   FormatMSG(GetLastError(), &lpszMsg);
   throw lpszMsg;
  }

  // Set the PropChangeParams structure.
  PropChangeParams.ClassInstallHeader.InstallFunction = DIF_PROPERTYCHANGE;
  PropChangeParams.Scope = DICS_FLAG_GLOBAL;
  PropChangeParams.StateChange = NewStatus;

  if (!SetupDiSetClassInstallParams(hDevInfo, &DeviceInfoData, (SP_CLASSINSTALL_HEADER *)&PropChangeParams,
sizeof(PropChangeParams)))
  {
   FormatMSG(GetLastError(), &lpszMsg);
   throw lpszMsg;
  }

  // Call the ClassInstaller and perform the change.
  if (!SetupDiCallClassInstaller(DIF_PROPERTYCHANGE,hDevInfo,&DeviceInfoData))
  {
   FormatMSG(GetLastError(), &lpszMsg);
   throw lpszMsg;
  }

  SetCursor(hCursor);
  return TRUE;
 }
 catch (TCHAR * pszError)
 {
  SetCursor(hCursor);
  ::MessageBox(NULL,pszError,_T("提示"),MB_OK);
  if (NULL != lpszMsg)
  {
   LocalFree((HLOCAL)lpszMsg);
  }
  return FALSE;
 }
}

// 这些设备的启用和禁用主要有此函数实现
// 参数说明,nStatus 可取3个值,-1 :啥也不做
// 0 禁用设备,1启用设备。
// nIndex用于控制是对光驱,软驱还是对USB启用或禁用
// 也是可取3个值,0 代表软驱,1 代表光驱
// 2 代表USB
// 譬如要对光驱实现禁用,可以这样调用此函数
// ControlDisk(0, 1);
BOOL ControlDisk(int nStatus, int nIndex)
{
 if (-1 == nStatus)
 { 
  return FALSE;
 }

 LPTSTR lpszMsg = NULL;
 try
 {
  TCHAR * GUIDString = NULL;
  GUID guid;
  ZeroMemory(&guid, sizeof(GUID));
  switch(nIndex)
  {
   case 0: // 0 代表软驱
    GUIDString = _T("4D36E980-E325-11CE-BFC1-08002BE10318");
    UuidFromString((unsigned char *)GUIDString, &guid);
    break;
   case 1: // 1 代表光驱
    GUIDString = _T("4D36E965-E325-11CE-BFC1-08002BE10318");
    UuidFromString((unsigned char *)GUIDString, &guid);
    break;
   case 2: // 2 代表USB
    GUIDString = _T("36FC9E60-C465-11CF-8056-444553540000");
    UuidFromString((unsigned char *)GUIDString, &guid);
    break;
  }

  hDevInfo = SetupDiGetClassDevs(&guid,NULL,NULL,DIGCF_PRESENT);
  if (INVALID_HANDLE_VALUE == hDevInfo)
  {
   FormatMSG(GetLastError(), &lpszMsg);
   throw lpszMsg;
  }

  DWORD i;
  SP_DEVINFO_DATA DeviceInfoData;
  ZeroMemory(&DeviceInfoData, sizeof(SP_DEVINFO_DATA));
  DeviceInfoData.cbSize = sizeof(SP_DEVINFO_DATA);

  for (i = 0; SetupDiEnumDeviceInfo(hDevInfo, i, &DeviceInfoData); ++i)
  {
   if (1 == nStatus)
   {
    StateChange(DICS_ENABLE, i, hDevInfo);
   }
   else if (0 == nStatus)
   {
    StateChange(DICS_DISABLE, i, hDevInfo);
   }
  }

  // 释放 device information set
  return SetupDiDestroyDeviceInfoList(hDevInfo);
 }
 catch (TCHAR * pszError)
 {
  ::MessageBox(NULL,pszError,_T("提示"),MB_OK);
  if (NULL != lpszMsg)
  {
   LocalFree((HLOCAL)lpszMsg);
  }
  return FALSE;
 }
 return FALSE;
}

  经测试这样对光驱,软驱和USB实现禁用没有问题,但是当禁用过之后如果要对USB实现启用必须两次调用ControlDisk(1, 2);这个函数才可,其余的两个启用没什么问题。还有要说的就是我不是用的枚举所有的设备,然后再过滤(网上的那个禁用网卡的就是这种方法)。

  据我观察,在注册表的此项下下:

HKEY_LOCAL_MACHINE/SYSTEM/CurrentControlSet/Control/Class/

  有很多GUID,每个GUID代表一个设备,譬如这个是USB的

{36FC9E60-C465-11CF-8056-444553540000}

  这个是CDROM

{4D36E965-E325-11CE-BFC1-08002BE10318}

  等等了。

  这些值你即使把它改为别的GUID,我使用原来的GUID仍然可以实现我的功能。

相关文章推荐

用组策略彻低禁止USB存储设备、光驱、软驱、ZIP软驱

用组策略彻低禁止USB存储设备、光驱、软驱、ZIP软驱 先下载附件里的usb.adm文件 详细操作如下: 1,在DC里的OU里新建一条GPO组策略 2、添加至...

linux下挂载光驱、软驱、U盘

linux下挂载光驱、软驱、U盘   2008-02-28 21:46:26|  分类: Linux |  标签: |字号大中小 订阅 mount   -t&#...

WINPE 无光驱 无软驱 无引导区```重新搞掂操作系统``

Winpe安装使用图文详细教程             转至在此感谢深度给我带来了技术上的进步WinPE是一个小巧的winxp操作系统,利用WinPE我们可以实现无光驱、...

无光驱、软驱,通过PXE安装Windows 2003(转载)

http://www.baowei.org/blog/post-243.html 情景: 买了一台微型PC做二奶机,无光驱软驱,又懒得从主PC上拆下来用,于是打算通过网络安装的方式来装...

安装系统不求人 没有光驱和软驱也能行(7种方法)

安装系统不求人 就算没有光驱和软驱也能行(7种方法)微软的操作系统在全球的普及率是相当高的,在中国同样如此,以往操作系统都是放在光盘上的,所以安装系统时理所当然需要使用光驱,但由于网络的飞速发展,光驱...

Linux下的虚拟光驱和虚拟软驱

转载:http://my.oschina.net/orion/blog/28016  如果你是刚从Windows转到Linux的话,你可能要问Windows下面有虚拟光驱,Linux...

LINUX挂载光驱,软驱,U盘

linux下挂载光驱、软驱、usb   mount   -t   iso9660   /dev/cdrom   /mnt/cdrom    ...

Linux下的虚拟光驱和虚拟软驱

如果你是刚从Windows转到Linux的话,你可能要问Windows下面有虚拟光驱,Linux下面有没有呢?一般来说Windows能做的事情,Linux都能做到,并且做得更好。但是概念上面可能和Wi...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)