示例:Linux應用程序遍歷當前系統的PCI設備

     使用應用程序遍歷設備上的所有PCI設備,通過遍歷這些設備,可以獲取到每個設備對應的bus號,dev號,func號,以及每個PCI設備的額vendorID和deviceID

     示例程序中的iopl, outl, inl是i386架構的Linux或Unix系統調用,因此示例適用平臺有限。

     程序的功能類似於lspci命令

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<sys/io.h>
#define PCI_MAX_BUS 255
#define PCI_MAX_DEV 31
#define PCI_MAX_FUN 7
#define PCI_BASE_ADDR 0x80000000L
#define CONFIG_ADDR 0xcf8
#define CONFIG_DATA 0xcfc
typedef unsigned long DWORD;
typedef unsigned int WORD;

int main()
{
    WORD bus, dev, fun;
    DWORD addr, data;
    printf("\nbus#\tdev#\tfun#\tvendor#\t\tdevice#\n");
    // 改變當前I/O端口的權能級別
    if ( iopl(3) < 0 )
    {
        printf("iopl set error\n");
        return -1;
    }
    for (bus = 0; bus <= PCI_MAX_BUS; bus++)
        for (dev = 0; dev <= PCI_MAX_DEV; dev++)
            for (fun = 0; fun <= PCI_MAX_FUN; fun++)
            {
                addr = PCI_BASE_ADDR | (bus << 16) | (dev << 11) | (fun << 8);
                outl(addr, CONFIG_ADDR);
                data = inl(CONFIG_DATA);
                if (((data & 0xFFFF) != 0xFFFF) && (data != 0))
                {
                    // bus, dev, fun
                    printf("%02d \t%02d \t%02d \t", bus, dev, fun);
                    // vendorID、deviceID
                    printf("%04x \t\t%04x\n", (data & 0xFFFF), (data & 0xFFFF0000) >> 16);
                }
            }
    if (iopl(0) < 0 )
    {
        printf("iopl set error\n");
        return -1;
    }
    return 0;
}

注:示例中的系統調用以及關鍵寄存器地址,可以自行搜索,瞭解。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章