从文件句柄获得全路径

从文件句柄获得全路径这个问题,似乎是个“老大难”问题。
很久以前我就在水木清华见到过。最近又不断有人提到。
其实问题并不难,只是解决办法有点绕,不是调用一个API就能解决的。

问题的关键在于,形如”X:”的Dos设备名都是符号链接(SymblicLink),而文件打开后文件对象中保存的是逻辑卷设备名(如”\Device\HarddiskVolumeX”)。前者可以转换成后者,而后者却不能简单地转换成前者。以至于从句柄得到的路径总是“缺少”盘符这一部分。实际上,把所有的”X:”都变成设备名去匹配路径就可以了。

下面是演示代码,很简单,所以就不加注释啦 ^_^

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
#include <windows.h>
#include <ntsecapi.h>

#pragma comment (lib,"ntdll.lib")       // Copy From DDK

NTSYSAPI
NTSTATUS
NTAPI
ZwQueryObject(
    IN HANDLE ObjectHandle,
    IN ULONG ObjectInformationClass,
    OUT PVOID ObjectInformation,
    IN ULONG ObjectInformationLength,
    OUT PULONG ReturnLength OPTIONAL
    );

BOOL GetPathByHandle(HANDLE hFile, LPWSTR lpBuf, DWORD nBuf)
{
    ULONG m, n;
    WCHAR lpPath[MAX_PATH+4];
    WCHAR lpDrive[MAX_PATH];
    WCHAR lpDevName[MAX_PATH];
    if (ZwQueryObject(hFile, 1, lpPath, MAX_PATH+4, &m) >= 0 &&
        (m = GetLogicalDriveStringsW(MAX_PATH, lpDrive)) && m < MAX_PATH)
    {
        WCHAR *p = lpDrive;
        while (m = wcslen(p))
        {
            p[m-1] = L'\0';
            n = QueryDosDeviceW(p, lpDevName, MAX_PATH);
            if (n && n < MAX_PATH)
            {
                n = wcslen(lpDevName);
                if (!wcsnicmp(lpPath+4, lpDevName, n))
                {
                    wcsncpy(lpBuf, p, nBuf);
                    if (nBuf > 2) wcsncpy(lpBuf+2, lpPath+4+n, nBuf-2);
                    return TRUE;
                }
            }
            p += m + 1;
        }
    }
    return FALSE;
}

void main()
{
   WCHAR buf[MAX_PATH];
   HANDLE hFile = cr&#101;ateFile("C:\\boot.ini", 0, 0, 0, OPEN_EXISTING, 0, 0);
   if (hFile != INVALID_HANDLE_VALUE)
   {
       GetPathByHandle(hFile, buf, MAX_PATH);
       printf("%ws\n", buf);
       CloseHandle(hFile);
   }
   else
   {
       printf("createFile Failed: %d\n", GetLastError());
   }
}