程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 更多編程語言 >> 匯編語言 >> 實戰DeviceIoControl系列之三:制作磁盤鏡像文件

實戰DeviceIoControl系列之三:制作磁盤鏡像文件

編輯:匯編語言

Q DOS命令 DISKCOPY 給我很深的印象,現在也有許多“克隆”軟件,可以對磁盤進行全 盤復制。我想,要制作磁盤鏡像文件,DeviceIoControl 應該很有用武之地吧?

A 是的。這裡舉一個制作軟盤鏡像文件,功能類似於“DISKCOPY”的例子。

本例實現其功能的核心代碼如下:

// 打開磁盤
HANDLE OpenDisk(LPCTSTR filename)
{
   HANDLE hDisk;
   // 打開設備
   hDisk = ::CreateFile(filename,      // 文件名
    GENERIC_READ | GENERIC_WRITE,    // 讀寫方式
     FILE_SHARE_READ | FILE_SHARE_WRITE, // 共享方式
     NULL,                // 默認的安全描述符
     OPEN_EXISTING,            // 創建方式
     0,                  // 不需設置文件屬性
     NULL);                // 不需參照模板文件
   return hDisk;
}
// 獲取磁盤參數
BOOL GetDiskGeometry(HANDLE hDisk, PDISK_GEOMETRY lpGeometry)
{
   DWORD dwOutBytes;
   BOOL bResult;
   // 用IOCTL_DISK_GET_DRIVE_GEOMETRY 取磁盤參數
   bResult = ::DeviceIoControl(hDisk,    // 設備句柄
     IOCTL_DISK_GET_DRIVE_GEOMETRY,    // 取磁盤參數
     NULL, 0,               // 不需要輸入數據
     lpGeometry, sizeof(DISK_GEOMETRY),  // 輸出數據緩沖區
     &dwOutBytes,             // 輸出數據長度
     (LPOVERLAPPED)NULL);         // 用同步I/O
   return bResult;
}
// 從指定磁道開始讀磁盤
BOOL ReadTracks(HANDLE hDisk, PDISK_GEOMETRY lpGeometry, LPVOID pBuf, DWORD dwStartCylinder, DWORD dwCylinderNumber)
{
   DWORD VirtBufSize;
   DWORD BytesRead;
   // 大小
   VirtBufSize = lpGeometry->TracksPerCylinder * lpGeometry->SectorsPerTrack * lpGeometry->BytesPerSector;
  // 偏移
   ::SetFilePointer(hDisk, VirtBufSize*dwStartCylinder, NULL, FILE_BEGIN);
  return ::ReadFile(hDisk, pBuf, VirtBufSize*dwCylinderNumber, &BytesRead, NULL);
}
 // 從指定磁道開始寫磁盤
BOOL WriteTracks(HANDLE hDisk, PDISK_GEOMETRY lpGeometry, LPVOID pBuf, DWORD dwStartCylinder, DWORD
dwCylinderNumber)
{
   DWORD VirtBufSize;
   DWORD BytesWritten;
  // 大小
   VirtBufSize = lpGeometry->TracksPerCylinder * lpGeometry->SectorsPerTrack * lpGeometry->BytesPerSector;
  // 偏移
   ::SetFilePointer(hDisk, VirtBufSize*dwStartCylinder, NULL, FILE_BEGIN);
  return ::WriteFile(hDisk, pBuf, VirtBufSize*dwCylinderNumber, &BytesWritten, NULL);
}
// 從指定磁道開始格式化磁盤
BOOL LowLevelFormatTracks(HANDLE hDisk, PDISK_GEOMETRY lpGeometry, DWORD dwStartCylinder, DWORD dwCylinderNumber)
{
   FORMAT_PARAMETERS FormatParameters;
   PBAD_TRACK_NUMBER lpBadTrack;
   DWORD dwOutBytes;
   DWORD dwBufSize; 
   BOOL bResult;
   FormatParameters.MediaType = lpGeometry->MediaType;
   FormatParameters.StartCylinderNumber = dwStartCylinder;
   FormatParameters.EndCylinderNumber = dwStartCylinder + dwCylinderNumber - 1;
   FormatParameters.StartHeadNumber = 0;
   FormatParameters.EndHeadNumber = lpGeometry->TracksPerCylinder - 1;
   dwBufSize = lpGeometry->TracksPerCylinder * sizeof(BAD_TRACK_NUMBER);
   lpBadTrack = (PBAD_TRACK_NUMBER) new BYTE[dwBufSize];
   // 用IOCTL_DISK_FORMAT_TRACKS 對連續磁道進行低級格式化
   bResult = ::DeviceIoControl(hDisk,        // 設備句柄
     IOCTL_DISK_FORMAT_TRACKS,          // 低級格式化
     &FormatParameters, sizeof(FormatParameters), // 輸入數據緩沖區
     lpBadTrack, dwBufSize,            // 輸出數據緩沖區
     &dwOutBytes,                 // 輸出數據長度
     (LPOVERLAPPED)NULL);             // 用同步I/O
   delete lpBadTrack;
   return bResult;
}
// 將卷鎖定
BOOL LockVolume(HANDLE hDisk)
{
   DWORD dwOutBytes;
   BOOL bResult;
  // 用FSCTL_LOCK_VOLUME 鎖卷
   bResult = ::DeviceIoControl(hDisk,    // 設備句柄
     FSCTL_LOCK_VOLUME,          // 鎖卷
     NULL, 0,               // 不需要輸入數據
     NULL, 0,               // 不需要輸出數據
     &dwOutBytes,             // 輸出數據長度
     (LPOVERLAPPED)NULL);         // 用同步I/O
   return bResult;
}
// 將卷解鎖
BOOL UnlockVolume(HANDLE hDisk)
{
  DWORD dwOutBytes;
  BOOL bResult;
   // 用FSCTL_UNLOCK_VOLUME 開卷鎖
   bResult = ::DeviceIoControl(hDisk,    // 設備句柄
     FSCTL_UNLOCK_VOLUME,         // 開卷鎖
     NULL, 0,               // 不需要輸入數據
     NULL, 0,               // 不需要輸出數據
     &dwOutBytes,             // 輸出數據長度
     (LPOVERLAPPED)NULL);         // 用同步I/O
   return bResult;
}
// 將卷卸下
// 該操作使系統重新辨識磁盤,等效於重新插盤
BOOL DismountVolume(HANDLE hDisk)
{
  DWORD dwOutBytes;
  BOOL bResult;
  // 用FSCTL_DISMOUNT_VOLUME 卸卷
  bResult = ::DeviceIoControl(hDisk,    // 設備句柄
    FSCTL_DISMOUNT_VOLUME,        // 卸卷
    NULL, 0,               // 不需要輸入數據
     NULL, 0,               // 不需要輸出數據
     &dwOutBytes,             // 輸出數據長度
     (LPOVERLAPPED)NULL);         // 用同步I/O
   return bResult;
}

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved