DeviceIoControl获取文件LCN

本文介绍了在Windows编程中,如何利用DeviceIoControl函数操作NTFS文件系统,特别是获取文件的逻辑簇号(LCN)。讨论了NTFS文件结构、簇流、VCN和LCN的概念,并提供了源码示例。
摘要由CSDN通过智能技术生成

在灾备项目组做CDP(Continual Data Protection),刚刚入门,最近看了下关于磁盘的操作,记录一下DeviceIoControl()。

NTFS中的文件结构

NTFS文件系统中,磁盘文件存储管理的最小单位是簇(Cluster)。一个簇由连续的若干个扇区构成。对于一个文件来说,我们将其在磁盘上占用的连续的簇称为簇流(Cluster Runs,不知道怎么翻译,有人称之为簇流,我也就这么叫吧),一个文件可以占用多个簇流。当文件的内容较小时,会与该文件的常驻属性一同放置在MFT的项中,每个MFT项的大小在引导扇区中被定义,但目前所有的NTFS文件系统的MFT项的大小均为1024Bytes。当文件内容较大时,就会占用额外的簇空间来存储数据,称之为非常驻属性。

非常驻属性被存储在簇流空间,簇流就是一组连续的簇,用起始簇号和长度表示一个簇流。例如,一个非常驻属性分成三部分,第一部分存储在簇号为30,31,32,33和34的簇中,那么该簇流的起始簇号就是30,流长度为5。第二部分存储在簇号为66和67的簇中,则该簇流起始簇号为66,流长度为2。第三部分存储在簇号为39~42的簇中,该簇流的起始簇号就是39,流长度为4。该属性一共占用了三个簇流,像下图这样(截图,侵删):


NTFS使用VCN(Virtual Cluster Number,虚拟簇号)和LCN(Logical Cluster Number,逻辑簇号)来描述文件或部分文件的位置。一个文件的VCN从0开始,是文件内容的内部编号,也就是说,将文件按照逻辑簇的大小进行划分,然后从0开始编号,就是所谓的VCN。而LCN就是文件系统的逻辑簇号,对于一个卷区来说,每一个簇的逻辑簇号是独一无二的。通过这种VCN和LCN的映射关系,我们就能在文件系统中定位文件。返回上面的例子,逻辑簇号30~34存储了文件内容的第一部分,对应的VCN为0~4,第二部分存储在逻辑簇号为66~67的簇中,其对应的VCN为5~6,第三部分存储在逻辑簇号39~42的簇中,对应的VCN为7~10。于是有下图(截图,侵删):

可以使用DeviceIOControl函数获取物理磁盘的个数。具体实现方法如下: 1. 打开一个物理磁盘设备句柄,调用CreateFile函数,设置参数为: 文件名:物理磁盘设备路径,例如"\\\\.\\PhysicalDrive0",数字0表示第一个物理磁盘。 访问模式:GENERIC_READ | GENERIC_WRITE 共享模式:FILE_SHARE_READ | FILE_SHARE_WRITE 安全性描述符:NULL 2. 调用DeviceIOControl函数,设置参数为: 设备句柄:上一步中返回的句柄 控制码:IOCTL_STORAGE_QUERY_PROPERTY 输入缓冲区:STORAGE_PROPERTY_QUERY结构体,其中的PropertyId字段设置为StorageDeviceProperty,QueryType字段设置为PropertyStandardQuery 输入缓冲区大小:sizeof(STORAGE_PROPERTY_QUERY) 输出缓冲区:存放设备属性信息的缓冲区,可以先设置为一定大小,如果返回的信息超过缓冲区大小,则继续调用DeviceIOControl函数,并增大缓冲区大小,直到返回的信息能够全部存放在缓冲区中 输出缓冲区大小:缓冲区大小 返回值:如果函数调用成功,则返回非零值,否则返回0 3. 解析返回的设备属性信息,判断是否为物理磁盘,并统计物理磁盘的个数。 4. 关闭设备句柄,调用CloseHandle函数。 参考代码如下: ```c++ #include <windows.h> #include <winioctl.h> #include <stdio.h> int main() { HANDLE hDevice = CreateFile("\\\\.\\PhysicalDrive0", GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, 0, NULL); if (hDevice == INVALID_HANDLE_VALUE) { printf("CreateFile failed, error code: %d\n", GetLastError()); return 0; } STORAGE_PROPERTY_QUERY query; query.PropertyId = StorageDeviceProperty; query.QueryType = PropertyStandardQuery; STORAGE_DESCRIPTOR_HEADER header; DWORD dwBytesReturned = 0; BOOL bRet = DeviceIoControl(hDevice, IOCTL_STORAGE_QUERY_PROPERTY, &query, sizeof(query), &header, sizeof(header), &dwBytesReturned, NULL); if (!bRet) { printf("DeviceIoControl failed, error code: %d\n", GetLastError()); CloseHandle(hDevice); return 0; } DWORD dwBufferSize = sizeof(STORAGE_DEVICE_DESCRIPTOR) + 512; PSTORAGE_DEVICE_DESCRIPTOR pDesc = (PSTORAGE_DEVICE_DESCRIPTOR)malloc(dwBufferSize); ZeroMemory(pDesc, dwBufferSize); pDesc->Size = dwBufferSize; bRet = DeviceIoControl(hDevice, IOCTL_STORAGE_QUERY_PROPERTY, &query, sizeof(query), pDesc, dwBufferSize, &dwBytesReturned, NULL); if (!bRet) { printf("DeviceIoControl failed, error code: %d\n", GetLastError()); CloseHandle(hDevice); free(pDesc); return 0; } if ((pDesc->DeviceType == DIRECT_ACCESS_DEVICE) && (pDesc->BusType == BusTypeSata)) { printf("This is a physical disk.\n"); } else { printf("This is not a physical disk.\n"); } CloseHandle(hDevice); free(pDesc); return 0; } ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值