之前因為修bug,有接觸到與PCI有關的Code,我用容易懂的方式把它紀錄下來。並以一個簡單的範例來教導大家如何去存取PCI Device。
原理
根據PCI Spec(我沒翻過),我們可以透過0xcf8(index register)設定要讀取的位置,設定完之後透過0xcfc(data register)去取得資料。
要設定給0xfc8的位置,擁有以下意義:
31 Enabled 30 Reserved 24 23 Bus 16 15 Device 11 10 Func 8 7 Reg 2 1 0 0 0
PS. PCI裝置都由Bus+Device+Func去辨別
範例
我們目標是去讀到ICH10 GPIO Base Address。 根據ICH10 Spec, GPIO Base Address在LPC Interface Bridge Registers中,LPC在Bus=0,Devices=31,Fun=0的位置,也就是0x8000F800。 而儲存GPIO Base Address在0x8000F800+0x48的位置。因此首先設定0xcf8為0x8000F800+0x48,接著再從0xcfc讀值出來。
在Linux下可以這樣做:
unsigned long index_reg_addr = 0xcf8; unsigned long data_reg_addr = 0xcfc; // Set address to pci configuration space unsigned long address = 0x8000f800+0x48; if(!ioperm(index_reg_addr , 4, 1)){ outl(index_reg_addr , address); } // Read data unsigned int *result; if(!ioperm(data_reg_addr, 4, 1)){ *result= inl(data_reg_addr); }
如果要讀不同的PCI Device出來,僅要跑一個for loop,從0x80000000開始,每一個interval加0x800即可。每一次所增加的0x800代表著Device加1。
在Windows上可以透過WinIO去存取IO Port。
留言
張貼留言