使用應用程序遍歷設備上的所有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;
}
注:示例中的系統調用以及關鍵寄存器地址,可以自行搜索,瞭解。