Linux下C如何调用PCI Lib函数
Linux下C如何調(diào)用PCI Lib函數(shù)
在Linux下,可以通過"setpci"和"setpci"命令來(lái)訪問PCI設(shè)備的配置空間,那么能否用程序來(lái)訪問PCI 配置空間呢?答案當(dāng)然是肯定的,linux下提供了多個(gè)pci庫(kù)以供應(yīng)用程序訪問。下面就以最常見的為例,從安裝、使用和編譯的角度分別進(jìn)行說明。
安裝
在centos中,用超級(jí)用戶權(quán)限,可用下面的命令查看到和pci訪問相關(guān)的庫(kù)包括:
libpciaccess.i686 : PCI access library
libpciaccess.x86_64 : PCI access library
libpciaccess-devel.i686 : PCI access library development package
libpciaccess-devel.x86_64 : PCI access library development package
pciutils.x86_64 : PCI bus related utilities
pciutils-devel.i686 : Linux PCI development library
pciutils-devel.x86_64 : Linux PCI development library
pciutils-devel-static.i686 : Linux PCI static library
pciutils-devel-static.x86_64 : Linux PCI static library
pciutils-libs.i686 : Linux PCI library
pciutils-libs.x86_64 : Linux PCI library
由于實(shí)際系統(tǒng)是64bit的,并且以靜態(tài)庫(kù)的形式進(jìn)行使用,可以用下面的命令安裝pci lib 庫(kù):?
?yum install libpciaccess.x86_64 libpciaccess-devel.x86_64 pciutils-devel-static.x86_64
安裝完后,可以看到相應(yīng)的頭文件已經(jīng)最/usr/include下面了:
[root@localhost include]# find ./ -name "*pci*"
./linux/pci.h
./linux/pci_regs.h
./linux/virtio_pci.h
./sys/pci.h
./pci
./pci/pci.h
./pciaccess.h
并且靜態(tài)庫(kù)也已經(jīng)存在了:
[root@localhost lib64]# find ./ -name "*pci*"
./kde4/kcm_pci.so
./pkgconfig/libpci.pc
./pkgconfig/pciaccess.pc
./libpci.so.3
./libpci.so.3.2.1
./libpciaccess.so.0
./libpciaccess.so.0.11.1
./libpci.so
./libpci.a
./libpciaccess.so
通過上面的輸出可以看到,libpciaccess只有動(dòng)態(tài)庫(kù)的形式,而libpci既有動(dòng)態(tài)庫(kù)也有靜態(tài)庫(kù),這個(gè)和我們輸入的" yum install libpciaccess.x86_64 libpciaccess-devel.x86_64 pciutils-devel-static.x86_64"命令是吻合的。
使用
打開/usr/include/pci/pci.h,可以找到我們需要的函數(shù):
u8 pci_read_byte(struct pci_dev *, int pos) PCI_ABI; /* Access to configuration space */
u16 pci_read_word(struct pci_dev *, int pos) PCI_ABI;
u32 pci_read_long(struct pci_dev *, int pos) PCI_ABI;
int pci_read_block(struct pci_dev *, int pos, u8 *buf, int len) PCI_ABI;
int pci_read_vpd(struct pci_dev *d, int pos, u8 *buf, int len) PCI_ABI;
int pci_write_byte(struct pci_dev *, int pos, u8 data) PCI_ABI;
int pci_write_word(struct pci_dev *, int pos, u16 data) PCI_ABI;
int pci_write_long(struct pci_dev *, int pos, u32 data) PCI_ABI;
int pci_write_block(struct pci_dev *, int pos, u8 *buf, int len) PCI_ABI;
打開/usr/include/pciaccess.h,也有可以訪問配置空間的函數(shù):
struct pci_io_handle *pci_device_open_io(struct pci_device *dev, pciaddr_t base,
???????????????????? pciaddr_t size);
struct pci_io_handle *pci_legacy_open_io(struct pci_device *dev, pciaddr_t base,
???????????????????? pciaddr_t size);
void pci_device_close_io(struct pci_device *dev, struct pci_io_handle *handle);
uint32_t pci_io_read32(struct pci_io_handle *handle, uint32_t reg);
uint16_t pci_io_read16(struct pci_io_handle *handle, uint32_t reg);
uint8_t pci_io_read8(struct pci_io_handle *handle, uint32_t reg);
void pci_io_write32(struct pci_io_handle *handle, uint32_t reg, uint32_t data);
void pci_io_write16(struct pci_io_handle *handle, uint32_t reg, uint16_t data);
void pci_io_write8(struct pci_io_handle *handle, uint32_t reg, uint8_t data);
從上面的兩組函數(shù)接口來(lái)看,可以看到pciutils需要初始化struct pci_dev,而pciasscess庫(kù)需要初始化struct pci_io_handle *handle。那么該如何使用pciutils靜態(tài)庫(kù)呢,通過繼續(xù)閱讀/usr/include/pci/pci.h,可以看到下面的函數(shù)聲明和注釋:
/* Initialize PCI access */
struct pci_access *pci_alloc(void) PCI_ABI;
void pci_init(struct pci_access *) PCI_ABI;
void pci_cleanup(struct pci_access *) PCI_ABI;
/* Scanning of devices */
void pci_scan_bus(struct pci_access *acc) PCI_ABI;
struct pci_dev *pci_get_dev(struct pci_access *acc, int domain, int bus, int dev, int func) PCI_ABI; /* Raw access to specified device */
void pci_free_dev(struct pci_dev *) PCI_ABI;
據(jù)此我們不難猜測(cè)出使用pciutils的代碼流程,并且根據(jù)頭文件說明實(shí)現(xiàn)一個(gè)測(cè)試代碼test_pcilib.c:
int main(void)
{
??????? int i = 0;
??????? int ret = 0;
??????? struct pci_access * myaccess;
??????? struct pci_dev * mydev;
??????? myaccess = pci_alloc();
??????? pci_init(myaccess);
??????? mydev = pci_get_dev(myaccess, 0, 6, 0, 0);
??????? for (i = 0; i < 256; i++) {
??????????????? ret = pci_read_byte(mydev, i);
??????????????? printf("%d: %02x\n", i, ret);
??????? }
??????? pci_free_dev(mydev);
??????? pci_cleanup(myaccess);
??????? return 0;
}
編譯
因?yàn)橐呀?jīng)知道上面的代碼依賴于libpci.a靜態(tài)庫(kù),為此可用用下面的gcc命令進(jìn)行簡(jiǎn)單編譯:
[root@localhost testcases]# gcc -c -o test_pci.o test_pcilib.c
[root@localhost testcases]# gcc -Wall -o test_pci test_pci.o /usr/lib64/libpci.a
[root@localhost testcases]# ./test_pci
0: b5
1: 10
2: a0
3: 87
4: 07
5: 04
6: 10
7: 00
8: ca
9: 00
10: 80
11: 06
12: 10
.....
本文轉(zhuǎn)自存儲(chǔ)之廚51CTO博客,原文鏈接:http://blog.51cto.com/xiamachao/1744433?,如需轉(zhuǎn)載請(qǐng)自行聯(lián)系原作者
總結(jié)
以上是生活随笔為你收集整理的Linux下C如何调用PCI Lib函数的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: ant的下载与安装——mybatis学习
- 下一篇: WPF实例秀——如何获取UI元素的图像