Microsoft MVP성태의 닷넷 이야기
닷넷: 2316. C# - Port I/O를 이용한 PCI Configuration Space 정보 열람 [링크 복사], [링크+제목 복사],
조회: 2929
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일

(시리즈 글이 4개 있습니다.)
디버깅 기술: 217. WinDbg - PCI 장치 열거
; https://www.sysnet.pe.kr/2/0/13873

닷넷: 2315. C# - PCI 장치 열거 (레지스트리, SetupAPI)
; https://www.sysnet.pe.kr/2/0/13877

닷넷: 2316. C# - Port I/O를 이용한 PCI Configuration Space 정보 열람
; https://www.sysnet.pe.kr/2/0/13881

닷넷: 2317. C# - Memory Mapped I/O를 이용한 PCI Configuration Space 정보 열람
; https://www.sysnet.pe.kr/2/0/13883




C# - Port I/O를 이용한 PCI Configuration Space 정보 열람

지난 글에서,

WinDbg - PCI 장치 열거
; https://www.sysnet.pe.kr/2/0/13873

"PCI Configuration Space" 정보를 Port I/O와 Memory Mapped I/O를 이용해 조회할 수 있다고 했는데요, 이들 중 Port I/O를 이용한 방법을 지난번에 만들어 둔 device driver의 도움을 받아,

커널 메모리를 읽고 쓰는 NT Legacy driver와 C# 클라이언트 프로그램
; https://www.sysnet.pe.kr/2/0/12104

C# 코드로 가져와 보겠습니다. ^^




Port를 이용한 PCI Configuration Space 영역을 읽는 방법은 다음과 같습니다.

// 우선 0xcf8 포트로 읽어낼 PCI 장치와 offset을 지정하고,
(port write), 0xcf8, 0x80000000 | (bus << 16) | (device << 11) | (function << 8) | offset

// 0xcfc 포트로 4바이트 데이터를 읽어옵니다.
(port read), 0xcfc

0xcf8 포트에 쓰는 데이터 형식은 지난 글에서도 이미지로 설명한 적이 있는데요,

pci_tree_3.png

// PCI Configuration Address Register에 쓸 데이트 포맷

bit 0~1: 00
bit 2~7: Register Offset
bit 8~10: Function Number
bit 11~15: Device Number
bit 16~23: Bus Number
bit 24~30: Reserved
bit 31: Enable Bit

Bus, Device, Function을 제외하고 Register Offset으로 할당된 비트가 6비트지만, 사실 0~1 비트도 Register offset 영역에 포함되므로 엄밀히는 8비트 영역의 주소를 지정할 수 있습니다. 즉, Port I/O를 이용한 방식에서는 CAM(Configuration Access Mechanism)의 256 바이트까지만 읽어올 수 있고 PCIe에서 확장된 4KB 영역 전체를 읽어낼 수는 없습니다.

게다가 0~1 비트가 무조건 0이라는 점에서 Register offset 주소 지정은 최소 4바이트 단위로 지정해야 한다는 것을 유의하시면 됩니다.




그리하여, PCI 장치마다 0 ~ 256 바이트만큼을 읽어낼 수 있다는 것은 알았고, Configuration Address Register의 나머지 영역에 해당하는 Bus, Device, Function (줄여서 BDF) 값을 알아내는 것이 남았는데요, 즉, 원하는 PCI 장치의 Configuration Space를 읽어오려면, 이들 값을 알고 있어야 합니다.

그래서 지난 글을 통해 PCI 장치를 열거하는 방법을 알아본 건데요,

C# - PCI 장치 열거 (레지스트리, SetupAPI)
; https://www.sysnet.pe.kr/2/0/13877

한 가지 아쉬운 점은, 레지스트리를 이용하든 / SetupAPI를 이용하든 BDF 값을 필드별로 정확히 알아낼 수는 없고 문자열로부터 추출해야 한다는 것입니다.

가령 SetupAPI의 경우 지난 글에서는 SetupDiGetDeviceProperty를 이용해 DEVPROPKEY에 해당하는 값들을 구해왔는데요,

DEVPROPKEY structure
; https://learn.microsoft.com/en-us/windows-hardware/drivers/install/devpropkey

Property Keys
; https://learn.microsoft.com/en-us/windows-hardware/drivers/install/property-keys

"Unified Device Property Model Properties"로 나열되는 적절한 DEVPROPKEY 중에는 (제가 찾아본 한계로) Device, Function 값을 알 수 있는 키는 없었습니다. 대신, SetupDiGetDeviceRegistryProperty를 이용해 SPDRP_LOCATION_INFORMATION을 이용하면 다음과 같이 문자열로 추출하는 것이 가능합니다.

// SPDRP_LOCATION_INFORMATION으로 반환한 예 (문자열)

"PCI bus 0, device 2, function 0"

참고로, 단순히 BDF 값만 알아내는 것이라면 레지스트리를 이용하는 것도 좋은 방법입니다. 실제로 Computer\HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Enum\ 하위의 PCI 장치를 살펴보면 이런 식으로 LocationInformation 값이 주어지는데요,

@System32\drivers\pci.sys,#65536;PCI bus %1, device %2, function %3;(1,0,1)

마지막의 (1,0,1) 값이 각각 (Bus, Device, Function)을 나타냅니다.




이렇게 해서 해당 머신에 장착된 PCI 장치의 BDF 값을 열거했으면 이제 Port I/O를 이용해 Configuration Space를 읽어오는 것은 어렵지 않습니다.

static unsafe void Main(string[] args)
{
    using (KernelMemoryIO portIo = new KernelMemoryIO())
    {
        if (portIo.IsInitialized == false)
        {
            Console.WriteLine("Failed to open device");
            return;
        }

        foreach (BDF bdf in PciHelper.EnumeatePCI())
        {
            Console.WriteLine($"Bus: {bdf.Bus}, Device: {bdf.Device}, Function: {bdf.Function}");

            List<byte> bytes = new List<byte>();
            for (uint where = 0; where < 256; where += 4)
            {
                uint configCommand = CONFIG_CMD(bdf.Bus, bdf.Device, bdf.Function, where);
                portIo.Outportl(0xcf8, configCommand);

                uint data = portIo.Inportl(0xcfc, out int errorNo);

                if (errorNo != 0)
                {
                    Console.WriteLine($"Error: {errorNo}");
                    break;
                }
                else
                {
                    byte[] buffer = BitConverter.GetBytes(data);
                    bytes.AddRange(buffer);
                }
            }

            for (int i = 0; i < bytes.Count; i++)
            {
                Console.Write($"{bytes[i]:x2} ");
                if (i % 16 == 15)
                {
                    Console.WriteLine();
                }
            }

            Console.WriteLine();
        }
    }
}

위의 프로그램을 Hyper-V Gen1 VM에서 실행하면 대충 이런 결과가 나옵니다.

c:\ex> ConsoleApp1.exe
Bus: 0, Device: 7, Function: 1
86 80 10 71 07 00 00 02 01 00 01 06 00 00 80 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
...[생략]...
00 00 00 00 00 00 00 00 30 0f 00 00 00 00 00 00

Bus: 0, Device: 7, Function: 0
86 80 10 71 07 00 00 02 01 00 01 06 00 00 80 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
...[생략]...
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00

Bus: 0, Device: 0, Function: 0
86 80 92 71 06 00 00 02 03 00 00 06 00 00 00 00
00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
...[생략]...
00 00 00 00 00 00 00 00 20 0f 00 00 00 00 00 00

참고로, 그것의 첫 64바이트는 포맷이 고정돼 있기 때문에,

pci_config_space_struct_1.png

이에 맞춰 구조체를 정의해 출력하면 좀 더 다듬어진 결과를 얻을 수 있습니다. ^^

byte[] configSpaces = bytes.ToArray();
fixed (byte* ptr = configSpaces)
{
    PCIConfigSpace pciInfo = Marshal.PtrToStructure<PCIConfigSpace>((IntPtr)ptr);
    Console.WriteLine($"VendorId: {pciInfo.VendorId:x4}");
    Console.WriteLine($"DeviceId: {pciInfo.DeviceId:x4}");
}

[System.Runtime.CompilerServices.InlineArray(3)]
public struct ClassCode
{
    private byte _element0; // public 접근을 허용하지만 실용적이지 않음
                            // 필드는 단 한 개만 정의할 수 있음
}

[StructLayout(LayoutKind.Sequential)]
public struct PCIConfigSpace
{
    public ushort VendorId;
    public ushort DeviceId;

    public ushort Command;
    public ushort Status;

    public byte RevisionId;
    public ClassCode ClassCode;

    public byte CacheLineS;
    public byte LatTimer;
    public byte HeaderType;
    public byte BIST;

    public uint BAR1;
    public uint BAR2;
    public uint BAR3;
    public uint BAR4;
    public uint BAR5;
    public uint BAR6;

    public uint CardbusCISPointer;

    public ushort SubsystemVendorId;
    public ushort SubsystemId;

    public uint ExpansionROMBaseAddress;

    public byte CapPointer;
    public byte Reserved1;
    public byte Reserved2;
    public byte Reserved3;

    public uint Reserved;

    public byte InterruptLine;
    public byte InterruptPin;
    public byte MinGnt;
    public byte MaxLat;
}

뭐, 이 정도면 (DDK 영역이 아닌) 사용자 모드의 소프트웨어 개발자에게는 차고 넘칠 정도의 정보를 가져온 것이니 굳이 ECAM 4KB 영역까지 읽어낼 필요는 없을 것 같습니다. ^^

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




검색하다 보니, PCI 장치 열거를 통해 해당 환경이 XBox인지 확인하는 코드가 있군요. ^^

PCI 버스 접근
; https://blog.naver.com/kgh2797/90063940615

/* Check for Xbox by identifying device at PCI 0:0:0, if it's
* 0x10de/0x02a5 then we're running on an Xbox */

WRITE_PORT_ULONG((ULONG*) 0xcf8, CONFIG_CMD(0, 0, 0));
PciId = READ_PORT_ULONG((ULONG*) 0xcfc);
if (0x02a510de == PciId)
{
    XboxMachInit(CmdLine);
}
else
{
    PcMachInit(CmdLine);
}

#define CONFIG_CMD(bus, dev_fn, where) \
 (0x80000000 | (((ULONG)(bus)) << 16) | (((dev_fn) & 0x1F) << 11) | (((dev_fn) & 0xE0) << 3) | ((where) & ~3))





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







[최초 등록일: ]
[최종 수정일: 2/12/2025]

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

비밀번호

댓글 작성자
 




... 136  137  138  139  140  141  142  143  [144]  145  146  147  148  149  150  ...
NoWriterDateCnt.TitleFile(s)
1454정성태5/31/201326284Java: 15. Java 7 Control Panel 실행시키는 방법
1453정성태5/22/201325329기타: 32. Microsoft FTP 사이트에 접속하는 방법
1452정성태5/21/201333037Windows: 73. TabProcGrowth 값 삭제 후 IE를 실행시키면 다시 복원되는 경우 [3]
1451정성태5/17/201331960Windows: 72. 윈도우 서버 2012 기초 사용법
1450정성태5/16/201322733오류 유형: 176. SQL10007N Message "0" could not be retrieved. Reason code: "3"
1449정성태5/15/201329842오류 유형: 175. SpeechRecognitionEngine 사용 시 오류 유형 2가지
1448정성태5/14/201324835VC++: 68. #pragma warning(disable: ...)로 오류 제어가 안된다면?
1447정성태5/3/201326519개발 환경 구성: 191. Debugging Tools for Windows 독립 설치 버전 [1]
1446정성태4/30/201327318.NET Framework: 368. Encoding 타입의 대체(fallback) 메카니즘 [1]
1445정성태4/26/201325536디버깅 기술: 54. NT 서비스의 Main 메서드 안에서 Process.GetProcessesByName 호출 시 멈춤 현상 [1]
1444정성태4/26/201329547기타: 31. Internet Explorer: 자바스크립트로 숨겨진 파일 다운로드 경로를 알아내는 방법 [1]
1443정성태4/24/201325226개발 환경 구성: 190. Azure PaaS 웹 응용 프로그램 배포 후 SMTP 서버 구성 [2]
1442정성태4/21/201328795기타: 30. 마이크로소프트 워드의 CPU 점유 현상으로 글자 입력이 느려졌다면? [1]
1441정성태4/21/201335401.NET Framework: 367. LargeAddressAware 옵션이 적용된 닷넷 32비트 프로세스의 가용 메모리 [14]
1440정성태4/19/201324130오류 유형: 174. dumpbin.exe 실행시 mspdb110.dll 로드 오류
1439정성태4/18/201327983VS.NET IDE: 76. Visual Studio 2012와 Itanium 빌드 옵션 [2]
1438정성태4/17/201327398.NET Framework: 366. 다른 프로세스에 환경 변수 설정하는 방법 - 두 번째 이야기 [1]파일 다운로드1
1437정성태4/17/201327620VC++: 67. CRT(C Runtime DLL: msvcr...dll)에 대한 의존성 제거
1436정성태4/17/201333004.NET Framework: 365. Local SYSTEM 권한으로 코드를 실행하는 방법파일 다운로드1
1435정성태4/15/201341889Windows: 71. ad-hoc 보다 더 편리한 "가상 Wifi" 를 이용한 인터넷 공유 [2]
1434정성태4/9/201323178오류 유형: 173. TFS 서버의 이벤트 로그 오류 - WebHost failed to process a request. Parameter name: certificate
1433정성태4/9/201323474개발 환경 구성: 189. TFS에 설치된 SharePoint 의 PowerShell 콘솔 띄우는 방법
1432정성태4/5/201324485오류 유형: 172. System.Web.PipelineModuleStepContainer.GetEventCount 에서 NullReferenceException 이 발생한다면?
1431정성태4/5/201325124기타: 29. 부팅 가능한 (외장) HDD를 기존 부팅 메뉴에 추가하는 방법
1430정성태4/4/201326991제니퍼 .NET: 23. 모바일용 웹 사이트에서 발생하는 응답 시간 지연 현상 [5]파일 다운로드1
1429정성태3/29/201323355개발 환경 구성: 188. SCOM 2012 - ASP.NET 모니터링 방법
... 136  137  138  139  140  141  142  143  [144]  145  146  147  148  149  150  ...