Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 2개 있습니다.)

C# - WMI 논리 디스크가 속한 물리 디스크의 정보를 얻는 방법

다음과 같은 질문이 있군요. ^^

드라이브 문자를 통해서 물리 디스크 명칭을 알아내고 싶습니다.
; https://www.sysnet.pe.kr/3/0/5803

윈도우에서 "drive"는 논리적인 디스크 정보에 해당합니다. 즉, "물리적인 하드 디스크"를 1개 구매해서, 거기에 3개의 파티션을 나눈 다음 각각 C, D, E 드라이브 문자를 할당할 수 있는데요, 그런 경우 다음과 같이 정리할 수 있습니다.

물리 디스크 0
    논리 디스크 0 - C
    논리 디스크 1 - D
    논리 디스크 2 - E

그러니까, 논리 디스크 자체에는 물리 디스크의 Serial number를 확인할 수 없습니다. 그것을 확인하기 위해서는 논리 디스크가 속한 물리 디스크를 먼저 구하는 선행 작업이 필요합니다.




결국 문제는, 논리 디스크(Win32_LogicalDisk)와 물리 디스크의 관계를 연결할 수 있는 접점이 필요한 건데요, 이에 대해서는 다음의 Q&A에서 잘 설명하고 있습니다.

WMI Association of LogicalDisk with DiskPartition
; https://stackoverflow.com/questions/48116174/wmi-association-of-logicaldisk-with-diskpartition

위의 글에서 설명하듯이, 논리 디스크 정보에 포함된 "VolumeSerialNumber" 고윳값이 Win32_DiskDriveToDiskPartition을 통해 물리 디스크 정보인 Win32_DiskDrive에 대한 교각 역할을 합니다.

따라서 다음과 같이 코딩을 하면,

using System.Management;
using System.Runtime.Versioning;
using ConsoleApp1;

[assembly: SupportedOSPlatform("windows")]

// Install-Package System.Management
internal class Program
{
    static void Main(string[] args)
    {
        List<Win32_DiskDrive> physicalDisks = GetPhysicalDiskList();

        foreach (Win32_LogicalDisk drive in GetLogicalDiskList())
        {
            Console.WriteLine($"{nameof(drive.DeviceID)}: {drive.DeviceID}");
            Console.WriteLine($"{nameof(drive.VolumeSerialNumber)}: {drive.VolumeSerialNumber}");

            drive.PhysicalDisk = physicalDisks.Find(disk => disk.Partitions.Find(partition => partition.VolumeSerialNumber == drive.VolumeSerialNumber) != null);
            if (drive.PhysicalDisk != null)
            {
                Console.WriteLine($"{nameof(drive.PhysicalDisk.SerialNumber)}: {drive.PhysicalDisk.SerialNumber}");
                Console.WriteLine($"{nameof(drive.PhysicalDisk.DeviceID)}: {drive.PhysicalDisk.DeviceID}");
            }

            Console.WriteLine();
        }
    }

    public static List<Win32_LogicalDisk> GetLogicalDiskList()
    {
        List<Win32_LogicalDisk> drives = new List<Win32_LogicalDisk>();

        try
        {
            ManagementObjectSearcher searcher = new ManagementObjectSearcher("root\\CIMV2", "SELECT * FROM Win32_LogicalDisk");

            foreach (ManagementObject queryObj in searcher.Get())
            {
                Win32_LogicalDisk item = new Win32_LogicalDisk();

                item.Caption = queryObj.GetAsString("Caption");
                item.Description = queryObj.GetAsString("Description");
                item.DeviceID = queryObj.GetAsString("DeviceID");
                item.Name = queryObj.GetAsString("Name");
                item.VolumeName = queryObj.GetAsString("VolumeName");
                item.VolumeSerialNumber = queryObj.GetAsString("VolumeSerialNumber");

                drives.Add(item);
            }
        }
        catch (ManagementException)
        {
        }

        return drives;
    }

    public static List<Win32_DiskDrive> GetPhysicalDiskList()
    {
        List<Win32_DiskDrive> drives = new List<Win32_DiskDrive>();

        try
        {
            ManagementObjectSearcher searcher = new ManagementObjectSearcher("root\\CIMV2", "SELECT * FROM Win32_DiskDrive");

            foreach (ManagementObject queryObj in searcher.Get())
            {
                Win32_DiskDrive item = new Win32_DiskDrive();

                item.DeviceID = queryObj.GetAsString("DeviceID");
                item.Caption = queryObj.GetAsString("Caption");
                item.Description = queryObj.GetAsString("Description");
                item.Index = queryObj.GetAsUInt32("Index");

                item.SerialNumber = queryObj.GetAsString("SerialNumber").Trim();
                item.PNPDeviceID = queryObj.GetAsString("PNPDeviceID");

                ListPartitions(item);
                drives.Add(item);
            }
        }
        catch (ManagementException)
        {
        }

        return drives;
    }

    // https://stackoverflow.com/questions/48116174/wmi-association-of-logicaldisk-with-diskpartition
    private static void ListPartitions(Win32_DiskDrive drive)
    {
        string assocQuery = $"Associators of {{Win32_DiskDrive.DeviceID='{drive.DeviceID}'}} where AssocClass=Win32_DiskDriveToDiskPartition";
        var assocPart = new ManagementObjectSearcher(assocQuery);
        assocPart.Options.Timeout = System.Management.EnumerationOptions.InfiniteTimeout;

        foreach (ManagementObject moPart in assocPart.Get())
        {
            string logDiskQuery = $"Associators of {{Win32_DiskPartition.DeviceID='{moPart.Properties["DeviceID"].Value.ToString()}'}} where AssocClass=Win32_LogicalDiskToPartition";

            var logDisk = new ManagementObjectSearcher(logDiskQuery);
            logDisk.Options.Timeout = System.Management.EnumerationOptions.InfiniteTimeout;

            foreach (var logDiskEnu in logDisk.Get())
            {
                Win32_DiskDriveToDiskPartition partition = new Win32_DiskDriveToDiskPartition();
                partition.VolumeSerialNumber = logDiskEnu.GetAsString("VolumeSerialNumber");

                drive.Partitions.Add(partition);
            }
        }
    }
}

public class Win32_LogicalDisk
{
    public string Caption = "";
    public string Description = "";
    public string DeviceID = "";
    public string Name = "";
    public string VolumeName = "";
    public string VolumeSerialNumber = "";

    public Win32_DiskDrive? PhysicalDisk;
}

public class Win32_DiskDrive
{
    public string DeviceID = "";
    public string Caption = "";
    public string Description = "";
    public uint Index;
    public string SerialNumber = "";
    public string PNPDeviceID = "";

    public List<Win32_DiskDriveToDiskPartition> Partitions = new List<Win32_DiskDriveToDiskPartition>();
}

public class Win32_DiskDriveToDiskPartition
{
    public string VolumeSerialNumber = "";
}

실행 시 이와 같은 결과가 나옵니다.

Ligical-DeviceID: C:
VolumeSerialNumber: C54B0E7F
SerialNumber: 0015_4222_153C_557E.
Physical-DeviceID: \\.\PHYSICALDRIVE1

Ligical-DeviceID: D:
VolumeSerialNumber: D29D0282
SerialNumber: 0015_4222_153C_5562.
Physical-DeviceID: \\.\PHYSICALDRIVE2

Ligical-DeviceID: E:
VolumeSerialNumber: A8A69CBD
SerialNumber: 0015_4222_153C_5562.
Physical-DeviceID: \\.\PHYSICALDRIVE2

Ligical-DeviceID: F:
VolumeSerialNumber: 38C805A0
SerialNumber: WD-WXT1DC0JKWHY
Physical-DeviceID: \\.\PHYSICALDRIVE0

실제로 해당 시스템에 설치된 물리 디스크에 따른 디스크 정보는 아래와 같은 상황입니다.

물리 디스크 - PHYSICALDRIVE0
    논리 디스크 F:

물리 디스크 - PHYSICALDRIVE1
    논리 디스크 C:

물리 디스크 - PHYSICALDRIVE2
    논리 디스크 D:
    논리 디스크 E:

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 1/24/2023]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2023-01-25 09시10분
[dsscrolls] 정말 감사합니다! 저 접점을 못 찾아서 헤매고 있었는데 큰 도움이 되었습니다.
[guest]
2023-01-25 06시33분
[dsscrolls] 델파이로 작성하려니 ManagementObjectSearcher 구현 등이 조금 어렵게 되어서 다른 방안을 찾아봤습니다.

1. Select Antecedent from Win32_LogicalDiskToPartition where Dependent="Win32_LogicalDisk.DeviceID="C:"" 로 Antecedent를 추출하고.

2. Antecedent를 Select * from Win32_DiskDriveToDiskPartition where Dependent="%s" 에 대입해서 물리 드라이브 이름(번호)를 얻고

3. Select SerialNumber from Win32_DiskDrive where DeviceID="\\\\.\\PHYSICALDRIVE%n" 에 번호를 대입하여 해당 물리 드라이브의 시리얼 번호 추출에 성공했습니다.

보여주신 글이 아니었더라면 이 방법을 못 떠올렸겠죠. 감사합니다.
[guest]

1  2  3  4  5  6  7  8  9  10  [11]  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13667정성태7/7/20246622닷넷: 2273. C# - 리눅스 환경에서의 Hyper-V Socket 연동 (AF_VSOCK)파일 다운로드1
13666정성태7/7/20247700Linux: 74. C++ - Vsock 예제 (Hyper-V Socket 연동)파일 다운로드1
13665정성태7/6/20247880Linux: 73. Linux 측의 socat을 이용한 Hyper-V 호스트와의 vsock 테스트파일 다운로드1
13663정성태7/5/20247474닷넷: 2272. C# - Hyper-V Socket 통신(AF_HYPERV, AF_VSOCK)의 VMID Wildcards 유형파일 다운로드1
13662정성태7/4/20247491닷넷: 2271. C# - WSL 2 VM의 VM ID를 알아내는 방법 - Host Compute System API파일 다운로드1
13661정성태7/3/20247413Linux: 72. g++ - 다른 버전의 GLIBC로 소스코드 빌드
13660정성태7/3/20247522오류 유형: 912. Visual C++ - Linux 프로젝트 빌드 오류
13659정성태7/1/20247858개발 환경 구성: 715. Windows - WSL 2 환경의 Docker Desktop 네트워크
13658정성태6/28/20248235개발 환경 구성: 714. WSL 2 인스턴스와 호스트 측의 Hyper-V에 운영 중인 VM과 네트워크 연결을 하는 방법 - 두 번째 이야기
13657정성태6/27/20247911닷넷: 2270. C# - Hyper-V Socket 통신(AF_HYPERV, AF_VSOCK)을 위한 EndPoint 사용자 정의
13656정성태6/27/20248085Windows: 264. WSL 2 VM의 swap 파일 위치
13655정성태6/24/20247850닷넷: 2269. C# - Win32 Resource 포맷 해석파일 다운로드1
13654정성태6/24/20247787오류 유형: 911. shutdown - The entered computer name is not valid or remote shutdown is not supported on the target computer.
13653정성태6/22/20247937닷넷: 2268. C# 코드에서 MAKEINTREOURCE 매크로 처리
13652정성태6/21/20249246닷넷: 2267. C# - Linux 환경에서 (Reflection 없이) DLL AssemblyFileVersion 구하는 방법파일 다운로드2
13651정성태6/19/20248487닷넷: 2266. C# - (Reflection 없이) DLL AssemblyFileVersion 구하는 방법파일 다운로드1
13650정성태6/18/20248410개발 환경 구성: 713. "WSL --debug-shell"로 살펴보는 WSL 2 VM의 리눅스 환경
13649정성태6/18/20247957오류 유형: 910. windbg - !py 확장 명령어 실행 시 "failed to find python interpreter" (2)
13648정성태6/17/20248278오류 유형: 909. C# - DynamicMethod 사용 시 System.TypeAccessException
13647정성태6/16/20249344개발 환경 구성: 712. Windows - WSL 2의 네트워크 통신 방법 - 세 번째 이야기 (같은 IP를 공유하는 WSL 2 인스턴스) [1]
13646정성태6/14/20247758오류 유형: 908. Process Explorer - "Error configuring dump resources: The system cannot find the file specified."
13645정성태6/13/20248197개발 환경 구성: 711. Visual Studio로 개발 시 기본 등록하는 dev tag 이미지로 Docker Desktop k8s에서 실행하는 방법
13644정성태6/12/20248868닷넷: 2265. C# - System.Text.Json의 기본적인 (한글 등에서의) escape 처리 [1]
13643정성태6/12/20248315오류 유형: 907. MySqlConnector 사용 시 System.IO.FileLoadException 오류
13642정성태6/11/20248198스크립트: 65. 파이썬 - asgi 버전(2, 3)에 따라 달라지는 uvicorn 호스팅
13641정성태6/11/20248673Linux: 71. Ubuntu 20.04를 22.04로 업데이트
1  2  3  4  5  6  7  8  9  10  [11]  12  13  14  15  ...