0

https://docs.microsoft.com/ko-kr/windows/win32/api/virtdisk/nf-virtdisk-querychangesvirtualdisk?redirectedfrom=MSDN

DWORD QueryChangesVirtualDisk(
  HANDLE                            VirtualDiskHandle,
  PCWSTR                            ChangeTrackingId,
  ULONG64                           ByteOffset,
  ULONG64                           ByteLength,
  QUERY_CHANGES_VIRTUAL_DISK_FLAG   Flags,
  PQUERY_CHANGES_VIRTUAL_DISK_RANGE Ranges,
  PULONG                            RangeCount,
  PULONG64                          ProcessedLength
);

我正在尝试获取两个虚拟磁盘或两个 Resillent Change Tracking ID (RCT) 之间的区别

RangeCount总是返回为 0

我用两个枚举参数打开了虚拟磁盘文件(vhdx)VIRTUAL_DISK_ACCESS_GET_INFOOPEN_VIRTUAL_DISK_FLAG_NONE

opStatus = OpenVirtualDisk(
        &storageType,
        VirtualDiskPath,
        VIRTUAL_DISK_ACCESS_GET_INFO,
        OPEN_VIRTUAL_DISK_FLAG_NONE,
        NULL,
        &vhdHandle ); // output handle

然后使用QueryChangesVirtualDisk()

opStatus = QueryChangesVirtualDisk(
    vhdHandle,
    ChangeTrackingId,
    0,
    32212254720,
    QUERY_CHANGES_VIRTUAL_DISK_FLAG_NONE,
    pRctRanges,
    &rctRangeCnt,
    &processedLength
    );

OpenVirtualDisk() QueryChangesVirtualDisk()

两个函数都没有返回错误,但在每种情况下RangeCount的值总是 0

虚拟磁盘文件和RCT ID肯定有区别

请给我一些建议

#include <windows.h>
#include <stdio.h>
#include <stdlib.h>
#include <initguid.h>
#include <strsafe.h>
#include <virtdisk.h>

#include "Storage.h"

DWORD QueryChangesVirtualDisk(_In_    LPCWSTR     VirtualDiskPath, _In_    LPCWSTR     ChangeTrackingId)
{
    VIRTUAL_STORAGE_TYPE storageType;
    PGET_VIRTUAL_DISK_INFO diskInfo;
    ULONG diskInfoSize;
    DWORD opStatus;

    HANDLE vhdHandle;

    QUERY_CHANGES_VIRTUAL_DISK_RANGE *pRctRanges;
    ULONG                            rctRangeCnt;
    ULONG64                          processedLength;
    UINT                             i;

    vhdHandle = INVALID_HANDLE_VALUE;
    diskInfo = NULL;
    diskInfoSize = sizeof(GET_VIRTUAL_DISK_INFO);

    rctRangeCnt = 0L;
    processedLength = 0L;
    pRctRanges = NULL;

    i = 0;

    diskInfo = (PGET_VIRTUAL_DISK_INFO)malloc(diskInfoSize);
    if (diskInfo == NULL)
    {
        opStatus = ERROR_NOT_ENOUGH_MEMORY;
        goto Cleanup;
    }

    storageType.DeviceId = VIRTUAL_STORAGE_TYPE_DEVICE_UNKNOWN;
    storageType.VendorId = VIRTUAL_STORAGE_TYPE_VENDOR_UNKNOWN;

    //////////////////////////////////////////////////////////////////

    opStatus = OpenVirtualDisk(
        &storageType,
        VirtualDiskPath,
        VIRTUAL_DISK_ACCESS_GET_INFO,
        OPEN_VIRTUAL_DISK_FLAG_NONE,
        NULL,
        &vhdHandle);

    if (opStatus != ERROR_SUCCESS)
    {
        wprintf(L"OpenVirtualDisk fail\n");
        goto Cleanup;
    }

    opStatus = QueryChangesVirtualDisk(
        vhdHandle,
        ChangeTrackingId,
        0,
        32212254720,
        QUERY_CHANGES_VIRTUAL_DISK_FLAG_NONE,
        pRctRanges,
        &rctRangeCnt,
        &processedLength
        );

    wprintf(L"rctRangeCnt : %lu\n", rctRangeCnt);
    wprintf(L"processedLength : %llu\n", processedLength);

    for (i = 0; i < rctRangeCnt; i++)
    {
        wprintf(L"ByteOffset : %lld   ByteLength : %lld\n", pRctRanges[i].ByteOffset, pRctRanges[i].ByteLength);
    }

    if (opStatus != ERROR_SUCCESS)
    {
        wprintf(L"QueryChangesVirtualDisk fail\n");
        goto Cleanup;
    }

Cleanup:

    if (opStatus == ERROR_SUCCESS)
    {
        wprintf(L"success\n");
    }
    else
    {
        wprintf(L"error = %u\n", opStatus);
    }

    if (vhdHandle != INVALID_HANDLE_VALUE)
    {
        CloseHandle(vhdHandle);
    }

    if (diskInfo != NULL)
    {
        free(diskInfo);
    }

    if (pRctRanges != NULL)
    {
        for (i = 0; i < rctRangeCnt; i++)
        {
            free(&pRctRanges[i]);
        }
    }

    return opStatus;
}
4

1 回答 1

1

问题是您为Ranges参数传递了一个空指针,而为RangeCount.

因为Ranges您应该将指针传递给第一个元素或QUERY_CHANGES_VIRTUAL_DISK_RANGE元素数组,并且RangeCount应该将其初始化为该数组中的元素数。

QueryChangesVirtualDisk函数返回时,它将被修改RangeCountRanges数组中已初始化元素的数量。

从您链接到的文档中,关于RangeCount论点:

输入时,该值指示参数指向QUERY_CHANGES_VIRTUAL_DISK_RANGE的数组可以容纳的结构数。Ranges输出时,该值包含该QUERY_CHANGES_VIRTUAL_DISK_RANGE方法放置在数组中的结构的数量。

简而言之:该函数不会为您创建此数组,您必须在调用该函数之前执行此操作。

于 2020-02-13T07:52:41.227 回答