PCI總線與接口技術(shù)
1、PCI總線的發(fā)展歷史
本文引用地址:http://cafeforensic.com/article/201612/328784.htm1990年,起源于Intel架構(gòu)開發(fā)實(shí)驗(yàn)室,全稱是Peripheral Component Interconnect,首先在服務(wù)器中使用,代替了原來的MCA以及EISA,EISA直到2000年才宣告退出歷史舞臺。直到1994年,才開始在PC上廣泛使用,代替了原來的VESA。在1995年的中期,蘋果電腦也開始使用PCI總線電氣規(guī)范。后來PCI也增加了許多新的功能,比如66MHz,3.3V標(biāo)準(zhǔn),以及133MHz的PCI-X。2004年,出現(xiàn)了PCI-Express,原來的PCI總線電氣規(guī)范被稱為了傳統(tǒng)PCI(Conventional PCI)。
2、PCI總線地址空間
獨(dú)立的內(nèi)存以及IO接口,由軟件分配。而另外一個地址空間-PCI配置空間(PCI Configuration Space),使用獨(dú)立的地址,允許軟件決定每一個連入的設(shè)備需要多少內(nèi)存以及IO地址空間。通過設(shè)備配置空間寄存器的設(shè)置,每一個設(shè)備至多可以申請6個內(nèi)存以及IO地址空間。PCI配置空間還包括了連入設(shè)備的相關(guān)信息,根據(jù)這些信息,操作系統(tǒng)可以方便地使用相應(yīng)的驅(qū)動來使用這些設(shè)備。為了使PCI總線公平地使用PCI總線,提供了一種等待計(jì)時的功能,計(jì)時器在設(shè)備獲得了總線使用權(quán)時以PCI時鐘信號的速率開始計(jì)時,直到減為零,設(shè)備釋放總線占有權(quán)。
3、關(guān)于PCI的中斷
四個中斷,屬于電平觸發(fā)方式(邊沿觸發(fā)方式的中斷容易丟失)。單功能設(shè)備只是使用INTA#,多功能設(shè)備使用多個中斷。四個中斷通過PCI橋(兩個PCI總線之間)映射到系統(tǒng)中斷上,所以軟件無從得知是那個PCI中斷。后來的PCI加入了消息信號中斷的機(jī)制,PCI-Express使用的也是消息信號中斷機(jī)制,而沒有物理中斷線。
4、PCI電氣規(guī)范
PCI卡的尺寸,長度為174.63mm,高度為36.068~106.68mm。一下是一幅PCI個引腳定義的圖。
5、PCI設(shè)備的初始化
機(jī)器上電的時候,配置軟件必須掃描PCI總線,確定有哪些PCI設(shè)備,然后加載相應(yīng)的驅(qū)動程序。所有PCI設(shè)備都必須實(shí)現(xiàn)PCI協(xié)議規(guī)定必須的配置寄存器。而對PCI配置訪問實(shí)際上就是訪問設(shè)備的配置寄存器。
6、PCI配置空間
配置空間各項(xiàng)數(shù)據(jù)說明:
廠商識別碼(Vendor ID):標(biāo)識設(shè)備的制造者,有PCI SIG來分配。0FFFH表示未配置任何設(shè)備。
設(shè)備識別碼(Device ID):標(biāo)識特定的設(shè)備,具體代碼由廠商分配。
版本ID,Revision ID:指定一個設(shè)備特有的版本號。
Class Code(分類代碼):用于設(shè)備分類。0BH處為基本分類代碼;0A處為子分類代碼;09H處為標(biāo)識一個專用的寄存器級編程接口,便于設(shè)備的軟件可以與設(shè)備交互數(shù)據(jù)。
命令寄存器(Command):為發(fā)出和響應(yīng)PCI總線命令提供了對設(shè)備粗略的控制。
狀態(tài)寄存器(Status):用于記錄PCI總線有關(guān)操作的狀態(tài)信息,系統(tǒng)對該寄存器的讀操作無特殊要求。
基地址寄存器(Base Address Registers):供地址映射使用,使PCI的IO映射以及內(nèi)存映射與具體設(shè)備無關(guān)。
擴(kuò)展ROM的基地址寄存器(Expansion ROM Base Address):用來處理那些配置了局部EPROM或者Flash ROM的基地址和大小。Cache大小寄存器:用來指定系統(tǒng)中Cache行的長度,每個參加Cache協(xié)議的設(shè)備都要使用該寄存器。
延時計(jì)時器:該寄存器以PCI總線時鐘為單位指定PCI總線主設(shè)備的延遲計(jì)時器。
內(nèi)含自測寄存器:可選的寄存器,用作內(nèi)含自測試的控制與狀態(tài)寄存器。
中斷引腳寄存器(Interrupt Pin):用來表示設(shè)備使用了哪個PCI中斷引腳。
中斷線寄存器(Interrupt Line):用來表示設(shè)備中的中斷引腳與系統(tǒng)可編程控制器8259的哪個中斷輸入線相連接。
MAX_GNT表示設(shè)備需要多長的突發(fā)傳輸時間。MAX_LAT表示對PCI總線進(jìn)行訪問的頻繁程度。
Card CIS Pointer:由在卡總線和PCI之間共享芯片的設(shè)備實(shí)現(xiàn)。
子系統(tǒng)廠商標(biāo)識和子系統(tǒng)標(biāo)識(Subsystem Vendor ID):用于惟一地標(biāo)識設(shè)備所駐留的插入卡和子系統(tǒng)。即插即用操作系統(tǒng)可以定位正確的驅(qū)動程序,裝載到存儲器。
7、PCI擴(kuò)展ROM
通過執(zhí)行擴(kuò)展ROM存放的代碼來完成與設(shè)備相關(guān)的初始化,同時也可能完成系統(tǒng)引導(dǎo)功能。該機(jī)制允許擴(kuò)展ROM中含有幾個不同的映像,以適應(yīng)不同的機(jī)器和處理器結(jié)構(gòu)。
凡是支持?jǐn)U展ROM的設(shè)備,必須支持按任意字節(jié)組合方式對ROM進(jìn)行訪問,特別強(qiáng)調(diào)的是要支持雙字(DWORD)訪問。擴(kuò)展ROM中的信息安排要與現(xiàn)有的適合于ISA和EISA以及MC適配器的Intel X86擴(kuò)展ROM中的頭標(biāo)區(qū)兼容。頭標(biāo)區(qū)中所給信息經(jīng)過了擴(kuò)充,從而使適配器的功能進(jìn)一步優(yōu)化使用,從而可以使擴(kuò)展ROM中的代碼在運(yùn)行期間所使用的存儲空間最小。
PCI擴(kuò)展ROM中代碼從不在原地執(zhí)行,而是將代碼從ROM中拷貝到RAM中執(zhí)行。這樣可以在初始化和運(yùn)行時動態(tài)地確定代碼長度,并且能夠改善代碼的執(zhí)行速度。
PCI對于不同的系統(tǒng)和處理器配置都應(yīng)該包含其編碼映像。每個映像由ROM首區(qū)(映像開始處)+數(shù)據(jù)配置區(qū)(映像的第64KB范圍內(nèi))組成。
ROM首區(qū)內(nèi)容
偏移 | 長度 | 值 | 說明 |
00H~01H | 2 | 55AAH | ROM標(biāo)簽字節(jié) |
02H~17H | 22 | XX | 保留 |
18H~19H | 2 | XX | 到PCI數(shù)據(jù)結(jié)構(gòu)指針 |
數(shù)據(jù)配置區(qū)
偏移量 | 長度 | 說明 | 偏移量 | 長度 | 說明 |
00H~03H | 4 | 標(biāo)簽,字符串"PCID" | 0DH~0FH | 3 | 分類代碼 |
04H~05H | 2 | 供應(yīng)商識別碼 | 10H~11H | 2 | 映像長度 |
06H~07H | 2 | 設(shè)備識別碼 | 12H~13H | 2 | 代碼數(shù)據(jù)的修改級別 |
08H~09H | 2 | 對重要產(chǎn)品數(shù)據(jù)的指針 | 14H~14H | 1 | 代碼類型 |
0AH~0BH | 2 | PCI數(shù)據(jù)結(jié)構(gòu)長度 | 15H~15H | 1 | 指示標(biāo)志 |
0CH~0CH | 1 | PCI數(shù)據(jù)結(jié)構(gòu)修改 | 16H~17H | 2 | 保留 |
8、關(guān)于PCI設(shè)備的初始化
系統(tǒng)POST首先檢查PCI設(shè)備在配置空間是否使用了擴(kuò)展ROM基地址寄存器(即是否有擴(kuò)展ROM),若使用了,POST將ROM映射到地址空間中一個未用的部分。
9、PCI BIOS
其主要作用有以下兩點(diǎn):
- 為應(yīng)用軟件或者PCI總線設(shè)備或者板卡提供服務(wù)調(diào)用。
- 初始化每個系統(tǒng)PCI設(shè)備。PCI BIOS輪流查詢每個PCI插槽,查找存在的PCI設(shè)備,讀取存在設(shè)備配置空間的頭標(biāo)區(qū),以決定設(shè)備的廠商號,類型和存儲需求等內(nèi)容。并且將分配的I/O或存儲空間地址回寫到每個設(shè)備配置空間的基地址寄存器中。
- PCI BIOS調(diào)用的入口以及返回值說明,對80x86機(jī)器,調(diào)用功能號為1AH,入口參數(shù)在AX中,返回值在AH中。下面列舉的是比較常用的,更多說明請參考PCI BIOS規(guī)則說明書。
功能說明 | 入口參數(shù)(AX) | 返回值說明 | 出口參數(shù)(AH) |
PCI BIOS存在查詢 | B101H | 成功調(diào)用 | 00H |
查找PCI 設(shè)備 | B102H | 不支持的功能 | 81H |
查找PCI 設(shè)備的類代碼 | B103H | 錯誤的廠商號 | 83H |
產(chǎn)生特殊周期 | B106H | 未找到設(shè)備 | 86H |
讀配置寄存器-單字節(jié)操作 | B108H | 錯誤的寄存器號 | 87H |
讀配置寄存器-單字操作 | B109H | 設(shè)置失敗 | 88H |
讀配置寄存器-雙字操作 | B10AH | 緩沖區(qū)太小 | 89H |
寫配置寄存器-單字操作 | B10BH | ||
寫配置寄存器-單字節(jié)操作 | B10CH | ||
寫配置寄存器-雙字節(jié)操作 | B10DH | ||
取得中斷線路選項(xiàng) | B10EH | ||
設(shè)置PCI中斷 | B10FH |
- 下面是一個關(guān)于通過PCI BIOS調(diào)用讀PCI配置寄存器的例子:
- .386
- ;FUNCTIONCODE
- PCI_FUNCTION_ID=0B1H
- PCI_BIOS_PRESENT=01H
- FIND_PCI_DEVICE=02H
- FIND_PCI_CLASS_CODE=03H
- GENERATE_SPECIAL_CYCLE=06H
- READ_CONFIG_BYTE=08H
- READ_CONFIG_WORD=09H
- READ_CONFIG_DWORD=0AH
- WRITE_CONFIG_BYTE=0BH
- WRITE_CONFIG_WORD=0CH
- WRITE_CONFIG_DWORD=0DH
- GET_IRQ_ROUTING_OPTIONS=0EH
- SET_PCI_IRQ=0FH
- ;RETURNCODE
- SUCCESSFUL=00H
- FUNC_NOT_SUPPORTED=81H
- BAD_VENDOR_ID=83H
- DEVICE_NOT_FOUND=86H
- BAD_REGISTER_NUMBER=87H
- SET_FAILED=88H
- BUFFER_TOO_SMALL=89H
- VID=0H
- DID=2H
- PCICMD=4H
- PCISTS=6H
- RID=8H
- CLCD=9H
- CALN=0CH
- LAT=0DH
- HDR=0EH
- BIST=0FH
- BADR0=10H
- BADR1=14H
- BADR2=18H
- BADR3=1CH
- BADR4=20H
- BADR5=24H
- EXPOM=30H
- INTLN=3CH
- INTPIN=3DH
- MINGNT=3EH
- MAXLAT=3FH
- SSTACKSEGMENTSTACKPARAUSE16
- DW64DUP(?)
- SSTACKENDS
- DATASEGMENTPARAUSE16
- MESDBPCICARDNOTFOUND!$
- MES0DB***********************PCICONFIGINFO*****************************,13,10,$
- MES1DBPCIBIOSNOTFOUND!,10,13,$
- MES2DBPCICONFIGREADERROR!$
- MES3DBVendorIdentification:10E8$
- MES4DBDeviceIdentification:5933$
- MES5DBPCICommandRegister:$
- MES6DBPCIStatusRegister:$
- MES7DBRevisionIdentificationRegister:$
- MES8DBClassCodeRegister:$
- MES9DBCacheLineSizeRegister:$
- MES10DBMasterLatencyTimer:$
- MES11DBHeaderType:$
- MES12DBBuilt-inSelf-test:$
- MES13DBBaseAddressRegister0:$
- MES14DBBaseAddressRegister1:$
- MES15DBBaseAddressRegister2:$
- MES16DBBaseAddressRegister3:$
- MES17DBBaseAddressRegister4:$
- MES18DBBaseAddressRegister5:$
- MES19DBExpansionRomBaseAddress:$
- MES20DBInterruptLine:$
- MES21DBInterruptPin:$
- MES22DBMinimumGrant:$
- MES23DBMaximumLatency:$
- BNDB?
- DN_FNDB?
- R_VALUEDD?
- V_VIDDW?
- V_DIDDW?
- V_PCICMDDW?
- V_PCISTSDW?
- V_RIDDB?
- V_CLCDDD?
- V_CALNDB?
- V_LATDB?
- V_HDRDB?
- V_BISTDB?
- V_BADR0DD?
- V_BADR1DD?
- V_BADR2DD?
- V_BADR3DD?
- V_BADR4DD?
- V_BADR5DD?
- V_EXPOMDD?
- V_INTLNDB?
- V_INTPINDB?
- V_MINGNTDB?
- V_MAXLATDB?
- DATAENDS
- CODESEGMENTPARAUSE16
- ASSUMECS:CODE,DS:DATA,SS:SSTACK
- START:MOVAX,DATA
- MOVDS,AX
- MOVAX,0B101H;查找PCIBIOS
- INT1AH
- JNCJUDGE1;如果CF被置位,則PCIBIOS不存在
- MOVDX,OFFSETMES1;顯示不存在信息
- MOVAH,09H
- INT21H
- JMPEXIT
- JUDGE1:CMPAH,00H
- JZJUDGE2;如果不等,則PCIBIOS不存在
- MOVDX,OFFSETMES1;顯示不存在信息
- MOVAH,09H
- INT21H
- JMPEXIT
- JUDGE2:CMPEDX,ICP;如果EDX中放的是"PCI"則說明PCIBIOS存在
- JZFIND
- MOVDX,OFFSETMES1;否則錯誤的設(shè)備
- MOVAH,09H
- INT21H
- JMPEXIT
- FIND:MOVAX,0B102H;找到了PCIBIOS,再查找指定PCI設(shè)備
- MOVCX,5933H;板卡的設(shè)備的ID
- MOVDX,10E8H;板卡的供應(yīng)商ID
- MOVSI,0;索引
- INT1AH
- JNCREAD
- MOVDX,OFFSETMES
- MOVAH,09H
- INT21H
- JMPEXIT
- READ:MOVBN,BH;保存總線號
- MOVDN_FN,BL;保存設(shè)備號
- CALLKENTER;回車換行
- MOVDX,OFFSETMES0
- MOVAH,09H
- INT21H
- CALLKENTER;回車換行
- MOVDX,OFFSETMES3;輸出供應(yīng)廠商ID
- MOVAH,09H
- INT21H
- CALLKENTER
- MOVDX,OFFSETMES4;輸出設(shè)備的ID
- MOVAH,09H
- INT21H
- CALLKENTER
- MOVAX,0B109H;讀命令寄存器,單字操作
- MOVBH,BN
- MOVBL,DN_FN
- MOVDI,PCICMD
- INT1AH
- JCERROR
- MOVDX,OFFSETMES5;顯示PCI命令寄存器內(nèi)容
- MOVAH,09H
- INT21H
- MOVAX,02H
- CALLSHOW
- CALLKENTER
- MOVAX,0B109H;讀PCI狀態(tài)寄存器內(nèi)容,單字操作
- MOVBH,BN
- MOVBL,DN_FN
- MOVDI,PCISTS
- INT1AH
- JCERROR
- MOVDX,OFFSETMES6;顯示狀態(tài)寄存器內(nèi)容
- MOVAH,09H
- INT21H
- MOVAX,02H
- CALLSHOW
- CALLKENTER
- MOVAX,0B108H;版本號,單字節(jié)操作
- MOVBH,BN
- MOVBL,DN_FN
- MOVDI,RID
- INT1AH
- JCERROR
- MOVDX,OFFSETMES7;顯示版本號
- MOVAH,09H
- INT21H
- MOVAX,01H
- CALLSHOW
- CALLKENTER
- MOVAX,0B108H;讀中斷引腳信號,單字節(jié)操作
- MOVBH,BN
- MOVBL,DN_FN
- MOVDI,INTLN
- INT1AH
- JCERROR
- MOVDX,OFFSETMES20;顯示中斷引腳
- MOVAH,09H
- INT21H
- MOVAX,01H
- CALLSHOW
- CALLKENTER
- MOVAX,0B10AH;讀配置寄存器,雙字操作
- MOVBH,BN;PCI設(shè)備的總線號
- MOVBL,DN_FN;設(shè)備以及功能號,入口參數(shù)
- MOVDI,BADR0
- INT1AH
- JCERROR
- MOVDX,OFFSETMES13;基址寄存器0
- MOVAH,09H
- INT21H
- MOVAX,04H
- CALLSHOW
- CALLKENTER
- MOVAX,0B10AH;讀配置寄存器,雙字操作
- MOVBH,BN;PCI設(shè)備的總線號
- MOVBL,DN_FN;設(shè)備及功能號,入口參數(shù)
- MOVDI,BADR1
- INT1AH
- JCERROR
- PUSHECX
- MOVDX,OFFSETMES14;基址寄存器1
- MOVAH,09H
- INT21H
- MOVAX,04H
- POPECX
- CALLSHOW
- CALLKENTER
- MOVAX,0B10AH;讀配置寄存器,雙字操作
- MOVBH,BN;PCI設(shè)備的總線號
- MOVBL,DN_FN;設(shè)備及功能號,入口參數(shù)
- MOVDI,BADR2
- INT1AH
- JCERROR
- MOVDX,OFFSETMES15;基地址寄存器2
- MOVAH,09H
- INT21H
- MOVAX,04H
- CALLSHOW
- CALLKENTER
- MOVAX,0B10AH;讀配置寄存器,雙字操作
- MOVBH,BN;PCI設(shè)備的總線號
- MOVBL,DN_FN;設(shè)備及功能號,入口參數(shù)
- MOVDI,BADR3
- INT1AH
- JCERROR
- MOVDX,OFFSETMES16;基地址寄存器3
- MOVAH,09H
- INT21H
- MOVAX,04H
- CALLSHOW
- CALLKENTER
- MOVAX,0B10AH;讀配置寄存器,雙字操作
- MOVBH,BN;PCI設(shè)備的總線號
- MOVBL,DN_FN;設(shè)備及功能號,入口參數(shù)
- MOVDI,BADR4
- INT1AH
- JCERROR
- MOVDX,OFFSETMES17;基地址寄存器4
- MOVAH,09H
- INT21H
- MOVAX,04H
- CALLSHOW
- JMPEXIT
- ERROR:CALLKENTER
- MOVDX,OFFSETMES2;顯示讀錯誤信息
- MOVAH,09H
- INT21H
- EXIT:MOVAH,4CH;返回DOS
- INT21H
- KENTERPROC
- MOVDL,0AH
- MOVAH,02H
- INT21H
- MOVDL,0DH
- MOVAH,02H
- INT21H
- RET
- KENTERENDP
- SHOWPROCNEAR;顯示子程序
- PUSHDX
- PUSHDI
- PUSHBX
- MOVDI,OFFSETR_VALUE
- MOV[DI],ECX;保存獲取的數(shù)據(jù)
- ADDDI,AX
- DECDI
- MOVCX,AX
- C1:MOVAL,[DI]
- PUSHAX
- SHRAL,4
- ANDAL,0FH;取高4位
- CMPAL,0AH;是否是A以上的數(shù)
- JBC2
- ADDAL,07H
- C2:ADDAL,30H
- MOVBH,AL
- POPAX
- ANDAL,0FH;取低4位
- CMPAL,0AH
- JBC3
- ADDAL,07H
- C3:ADDAL,30H
- MOVBL,AL
- MOVAH,2;顯示十六進(jìn)制數(shù)對應(yīng)的ACSII碼
- MOVDL,BH
- INT21H
- MOVDL,BL
- INT21H
- DECDI
- LOOPC1
- POPBX
- POPDI
- POPDX
- RET
- SHOWENDP
- CODEENDS
- ENDSTART
評論