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]

... 31  32  [33]  34  35  36  37  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12819정성태8/31/20218231.NET Framework: 1111. C# - FormattableString 타입
12818정성태8/31/20217463Windows: 198. 윈도우 - 작업 관리자에서 (tensorflow 등으로 인한) GPU 연산 부하 보는 방법
12817정성태8/31/202110101스크립트: 25. 파이썬 - 윈도우 환경에서 directml을 이용한 tensorflow의 AMD GPU 사용 방법
12816정성태8/30/202115451스크립트: 24. 파이썬 - tensorflow 2.6 NVidia GPU 사용 방법 [2]
12815정성태8/30/20218546개발 환경 구성: 602. WSL 2 - docker-desktop-data, docker-desktop (%LOCALAPPDATA%\Docker\wsl\data\ext4.vhdx) 파일을 다른 디렉터리로 옮기는 방법
12814정성태8/30/202110866.NET Framework: 1110. C# 11 - 인터페이스 내에 정적 추상 메서드 정의 가능 (DIM for Static Members) [2]파일 다운로드1
12813정성태8/29/20219062.NET Framework: 1109. C# 10 - (11) Lambda 개선파일 다운로드1
12812정성태8/28/20218686.NET Framework: 1108. C# 10 - (10) 개선된 #line 지시자
12811정성태8/27/20218919Linux: 44. 윈도우 개발자를 위한 리눅스 fork 동작 방식 설명 (파이썬 코드)
12810정성태8/27/20217723.NET Framework: 1107. .NET Core/5+에서 동적 컴파일한 C# 코드를 (Breakpoint도 활용하며) 디버깅하는 방법 - #line 지시자파일 다운로드1
12809정성태8/26/20218391.NET Framework: 1106. .NET Core/5+에서 C# 코드를 동적으로 컴파일/사용하는 방법 [1]파일 다운로드1
12808정성태8/25/20219624오류 유형: 758. go: ...: missing go.sum entry; to add it: go mod download ...
12807정성태8/25/20219605.NET Framework: 1105. C# 10 - (9) 비동기 메서드가 사용할 AsyncMethodBuilder 선택 가능파일 다운로드1
12806정성태8/24/20217245개발 환경 구성: 601. PyCharm - 다중 프로세스 디버깅 방법
12805정성태8/24/20218477.NET Framework: 1104. C# 10 - (8) 분해 구문에서 기존 변수의 재사용 가능파일 다운로드1
12804정성태8/24/20219188.NET Framework: 1103. C# 10 - (7) Source Generator V2 APIs
12803정성태8/23/20218922개발 환경 구성: 600. pip cache 디렉터리 옮기는 방법
12802정성태8/23/20219140.NET Framework: 1102. .NET Conf Mini 21.08 - WinUI 3 따라해 보기 [1]
12801정성태8/23/20218653.NET Framework: 1101. C# 10 - (6) record class 타입의 ToString 메서드를 sealed 처리 허용파일 다운로드1
12800정성태8/22/20218884개발 환경 구성: 599. PyCharm - (반대로) 원격 프로세스가 PyCharm에 디버그 연결하는 방법
12799정성태8/22/20218885.NET Framework: 1100. C# 10 - (5) 속성 패턴의 개선파일 다운로드1
12798정성태8/21/202110316개발 환경 구성: 598. PyCharm - 원격 프로세스를 디버그하는 방법
12797정성태8/21/20217982Windows: 197. TCP의 MSS(Maximum Segment Size) 크기는 고정된 것일까요?
12796정성태8/21/20218620.NET Framework: 1099. C# 10 - (4) 상수 문자열에 포맷 식 사용 가능파일 다운로드1
12795정성태8/20/20219239.NET Framework: 1098. .NET 6에 포함된 신규 BCL API - 스레드 관련
12794정성태8/20/20218724스크립트: 23. 파이썬 - WSGI를 만족하는 최소한의 구현 코드 및 PyCharm에서의 디버깅 방법 [1]
... 31  32  [33]  34  35  36  37  38  39  40  41  42  43  44  45  ...