示例: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;
}

注:示例中的系统调用以及关键寄存器地址,可以自行搜索,了解。

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