新聞中心

EEPW首頁(yè) > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > 關(guān)于ISA設(shè)備的驅(qū)動(dòng)程序轉(zhuǎn)換為PCI設(shè)備的Windows驅(qū)動(dòng)程序設(shè)計(jì)

關(guān)于ISA設(shè)備的驅(qū)動(dòng)程序轉(zhuǎn)換為PCI設(shè)備的Windows驅(qū)動(dòng)程序設(shè)計(jì)

作者: 時(shí)間:2006-05-07 來(lái)源:網(wǎng)絡(luò) 收藏

摘要:本文主要針對(duì)已經(jīng)研制好的ISA設(shè)備通過(guò)加上一個(gè)簡(jiǎn)單的PCI接口芯片便能正常工作在PCI模式下,實(shí)現(xiàn)由ISA擴(kuò)展板到PCI擴(kuò)展板的轉(zhuǎn)換。這時(shí)我們必須重新編寫(xiě)設(shè)備的驅(qū)動(dòng)程序才能使設(shè)備在Windows/Nt操作系統(tǒng)下正常工作。這里主要給出Windows下的解決方案和程序?qū)嵗?br>
關(guān)鍵詞:ISA設(shè)備驅(qū)動(dòng)程序 PCI設(shè)備驅(qū)動(dòng)程序 IRQ PCI配置空間

本文引用地址:http://butianyuan.cn/article/241611.htm

隨著計(jì)算機(jī)和通信技術(shù)的高速發(fā)展,ISA總線在速度、功能上已經(jīng)成為系統(tǒng)的瓶頸,而功能更強(qiáng)大的PCI總線成為首選。這時(shí)對(duì)現(xiàn)有的ISA設(shè)備稍加設(shè)計(jì)就可在PCI總線下工作就顯得非??傄?,但我們必須重新編寫(xiě)設(shè)備的驅(qū)動(dòng)程序才能使設(shè)備在Windows/Nt操作系統(tǒng)下正常工作。

開(kāi)發(fā)的驅(qū)動(dòng)程序是Win32 Drivers Mode(WDM)類型,使用的開(kāi)發(fā)工具是微軟提供的Device Driver Kit(DDK)和Vc++6.0,在進(jìn)行驅(qū)動(dòng)程序的調(diào)試時(shí)使用Numega公司的SoftIce產(chǎn)品。

1 .ISA設(shè)備與PCI設(shè)備的Windows驅(qū)動(dòng)程序的比較

(1).ISA設(shè)備與PCI設(shè)備驅(qū)動(dòng)程序獲得硬件資源途徑不同
一個(gè)ISA設(shè)備驅(qū)動(dòng)程序的資源是固定不變的,它是通過(guò).inf文件的[logconfig]節(jié)來(lái)配置的。如下面的例子:

[Xxx.LogConfig]

IOConfig=220-22f

IRQConfig=6

.inf中此節(jié)說(shuō)明此硬件資源的I/O地址范圍是220H到22FH,IRQ為6。

一個(gè)PCI設(shè)備驅(qū)動(dòng)程序的資源是操作系統(tǒng)自動(dòng)分配的,它是通過(guò)設(shè)備ID號(hào)和廠商ID號(hào)獲得設(shè)備的物理位置:總線號(hào)、器件號(hào)和功能號(hào),并利用它們尋址PCI配置空間,接著從配置空間獲得硬件資源。其中包括:中斷號(hào)、端口地址等。

(2).ISA設(shè)備與PCI設(shè)備驅(qū)動(dòng)程序?qū)χ袛嗵幚聿煌?br> 一個(gè)ISA設(shè)備驅(qū)動(dòng)程序的中斷模式可以是LevelSensitive也可以是Latched,而且中斷向量是否與其它設(shè)備共享都可以。

但是一個(gè)PCI設(shè)備驅(qū)動(dòng)程序的中斷模式必須是LevelSensitive,而且中斷向量必須是共享的。

(3).ISA設(shè)備與PCI設(shè)備驅(qū)動(dòng)程序安裝時(shí)需要編寫(xiě)的.inf文件不同

對(duì)于ISA設(shè)備,在安裝時(shí).inf文件必須有[logconfig]節(jié),而對(duì)于PCI設(shè)備,在安裝時(shí).inf文件必須有[Manufacturer]節(jié),來(lái)指明設(shè)備ID號(hào)和廠商ID號(hào),以便使硬件獲取系統(tǒng)資源。如:

[Manufacturer]

%PLX% = PLX.Mfg

[PLX.Mfg]

"PCI 9052RDK-860 Board" =DDInstall_9052,PCIVEN_10b5DEV_9050

其設(shè)備ID號(hào)為9050H,廠商ID號(hào)為10B5H。

2 驅(qū)動(dòng)程序的實(shí)現(xiàn)

通過(guò)上面的比較,我們知道只要對(duì)原有的ISA設(shè)備的驅(qū)動(dòng)程序的獲取資源部分作一定的改變,并在安裝時(shí)對(duì)inf文件進(jìn)行必要的修改就可以完成PCI模式的驅(qū)動(dòng)程序。

以下示例僅供參考。

NTSTATUS StartDevice(PDEVICE_OBJECT fdo,

PCM_PARTIAL_RESOURCE_LIST ResourceListRaw,

PCM_PARTIAL_RESOURCE_LIST ResourceListTranslated)

{

U32 i;

NTSTATUS status;

KIRQL irql; // interrupt level

KINTERRUPT_MODE mode; // interrupt mode

KAFFINITY affinity; // processor affinity for interrupt

PDEVICE_EXTENSION dx=(PDEVICE_EXTENSION)fdo->DeviceExtension;

PCI_COMMON_CONFIG pciRegs;

PCM_PARTIAL_RESOURCE_DESCRIPTOR ResourceRaw;

PCM_PARTIAL_RESOURCE_DESCRIPTOR ResourceTranslated;

ResourceRaw = ResourceListRaw->PartialDescriptors;

ResourceTranslated = ResourceListTranslated->PartialDescriptors;

// Read PCI Config Register

PciConfigRegisterBufferRead(

fdo,

pciRegs,

0,

sizeof(pciRegs)

);

for (i = 0; i ResourceListTranslated->Count;++i,++ResourceTranslated++ResourceRaw)

{

switch (ResourceTranslated->Type)

{

case CmResourceTypePort:

dx->PortStartAddress = ResourceTranslated->u.Port.Start;

dx->PortLength = ResourceTranslated->u.Port.Length;

dx->PortNeedsMapping = (ResourceTranslated->FlagsCM_RESOURCE_PORT_IO)==0;

break;

case CmResourceTypeInterrupt:

dx->InterruptIrql = (KIRQL)ResourceTranslated->u.Interrupt.Level;

dx->InterruptVector = ResourceTranslated->u.Interrupt.Vector; dx->

InterruptAffinity = ResourceTranslated->u.Interrupt.Affinity;

dx->InterruptMode = LevelSensitive;

dx->InterruptConnected = false;

break;

case CmResourceTypeNull:

case CmResourceTypeDma:

case CmResourceTypeDeviceSpecific:

case CmResourceTypeBusNumber:

// NonArbitrated ConfigData are currently #defined as the same number

case CmResourceTypeConfigData:

case CmResourceTypeDevicePrivate:

case CmResourceTypePcCardConfig:

case CmResourceTypeMfCardConfig:

//加入自己的代碼

break;

default:

break;

}

}


/* Device has been completely initialized and is ready to run. */

// Get the Vendor and Device ID

status = PciConfigRegisterBufferRead(

fdo,

i,

0,

sizeof(U32)

);

if (!NT_SUCCESS(status))

{

dx->Device.VendorId = 0xFFFF;

dx->Device.DeviceId = 0xFFFF;

}

else

{

// Record the Vendor and Device ID */

dx->Device.VendorId = i 0x0000FFFF;

dx->Device.DeviceId = i >> 16;

}

// Get the bus number and the slot number

status = GetBusSlotNumber(

dx->pPhysicalDeviceObject,

dx

);

if (!NT_SUCCESS(status))

{

return status;

}

return STATUS_SUCCESS;

}

共享中斷向量只需將IoConnectInterrupt函數(shù)的第九個(gè)參數(shù)值置為TRUE就可以。

實(shí)踐證明以上方法是可行的。

linux操作系統(tǒng)文章專題:linux操作系統(tǒng)詳解(linux不再難懂)


評(píng)論


相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉