Windows CE(EVC)下操作GPIO的方法——轉載

來源:互聯網
上載者:User
Windows CE下操作GPIO的方法
Windows CE下操作GPIO的方法
發布日期:2007-5-28 14:16:27   作者:   出處:

Windows CE下操作GPIO的方法(以ARM9 S3C2410為例)
  
 

GPIO 是ARM晶片最基本的輸入輸出通道,在ADS下操作就是一個單片機工作,直接讀寫其寄存器。在ARM9平台上,Windows CE系統將GPIO的真實位址(例如2410的GPIO的基地址為0x56000000)映射到虛擬位址空間(GPIO對應為0xB1600000),這樣,通過對這段虛擬位址空間的操作,就能夠完成對GPIO或者其他片內資源的控制、輸入輸出工作。
要操作一個平台的GPIO,在其對應BSP中按照基地址,找到虛擬位址,並且找到方便操作這個地址的資料結構就可以了,關鍵函數就是VirtualAlloc和VirtualCopy。並且CE的方便之處就是使用者態的應用程式仍然可以使用這兩個函數來訪問所有這些虛擬空間,對於不太複雜的程式,甚至可以省略寫驅動直接在應用程式中操作,其實在CE6之前,這些驅動也是工作在使用者態的。
下面以操作Samsung S3C2410的GPIO為例,講述這個步驟:
1.首先在BSP中的s2410.h檔案,找到虛擬位址映射以及操作GPIO的寄存器結構體(這個在自己製作一些特殊裝置的BSP時,會依據需要而發生更改)
//
// Registers : I/O port
//

#define IOP_BASE      0xB1600000 // 0x56000000
typedef struct  {
    unsigned int  rGPACON;  // 00
    unsigned int  rGPADAT;
    unsigned int  rPAD1[2];
  
    unsigned int  rGPBCON;  // 10
    unsigned int  rGPBDAT;
    unsigned int  rGPBUP;
    unsigned int  rPAD2;
  
    unsigned int  rGPCCON;  // 20
    unsigned int  rGPCDAT;
    unsigned int  rGPCUP;
    unsigned int  rPAD3;
  
    unsigned int  rGPDCON;  // 30
    unsigned int  rGPDDAT;
    unsigned int  rGPDUP;
    unsigned int  rPAD4;
  
    unsigned int  rGPECON;  // 40
    unsigned int  rGPEDAT;
    unsigned int  rGPEUP;
    unsigned int  rPAD5;
  
    unsigned int  rGPFCON;  // 50
    unsigned int  rGPFDAT;
    unsigned int  rGPFUP;
    unsigned int  rPAD6;
  
    unsigned int  rGPGCON;  // 60
    unsigned int  rGPGDAT;
    unsigned int  rGPGUP;
    unsigned int  rPAD7;
  
    unsigned int  rGPHCON;  // 70
    unsigned int  rGPHDAT;
    unsigned int  rGPHUP;
    unsigned int  rPAD8;
  
    unsigned int  rMISCCR;  // 80
    unsigned int  rDCKCON; 
    unsigned int  rEXTINT0;
    unsigned int  rEXTINT1; 
    unsigned int  rEXTINT2;  // 90
 unsigned int  rEINTFLT0;
 unsigned int  rEINTFLT1;
 unsigned int  rEINTFLT2;
 unsigned int  rEINTFLT3;  // A0
 unsigned int  rEINTMASK;
 unsigned int  rEINTPEND;
 unsigned int  rGSTATUS0;  // AC
 unsigned int  rGSTATUS1;  // B0
 unsigned int  rGSTATUS2;  // B4
 unsigned int  rGSTATUS3;  // B8
 unsigned int  rGSTATUS4;  // BC
 
}IOPreg; 
將這些複製備用。

2.在EVC中建立一個應用程式工程,由於VirtualCopy函數沒有在標頭檔中定義,但是在coredll.lib裡面提供了符號串連,所以我們這裡直接添加一個函數定義就OK了。
#ifdef __cplusplus
extern "C"
{
#endif

BOOL VirtualCopy( LPVOID, LPVOID, DWORD, DWORD );

#ifdef __cplusplus
}
#endif
同時將步驟1裡面的定義複製到這裡。

3.按照驅動程式裡面操作的方法在應用程式中寫GPIO操作函數
(1)定義一個寄存器結構體變數
volatile IOPreg *v_pIOPRegs;
(2)給這個變數分配空間並且映射到寄存器的空間上
v_pIOPRegs = (volatile IOPreg*)VirtualAlloc(0, sizeof(IOPreg), MEM_RESERVE, PAGE_NOACCESS);
 if (v_pIOPRegs == NULL)
 {
  DEBUGMSG (1,(TEXT("v_pIOPRegs is not allocated/n/r")));
  return TRUE;
 }
 if (!VirtualCopy((PVOID)v_pIOPRegs, (PVOID)IOP_BASE, sizeof(IOPreg), PAGE_READWRITE|PAGE_NOCACHE)) {
  DEBUGMSG (1,(TEXT("v_pIOPRegs is not mapped/n/r")));
  return TRUE;
 }
 DEBUGMSG (1,(TEXT("v_pIOPRegs is mapped to %x/n/r"), v_pIOPRegs));
這3個步驟之後,對v_pIOPRegs的操作將直接和GPIO的寄存器關聯
例如:設定GPB的控制寄存器為全部Output
v_pIOPRegs->rGPBCON=0x155555;
設定GPB的資料寄存器輸出高電平
v_pIOPRegs->rGPBDAT=0x3FF;

更多的操作,需要查閱ARM的datasheet以及WINCE的BSP源碼完成。

對於非ARM的平台,在CE下操作,也可以參考這個思路。

WINCE下I/O操作基礎
發布日期:2007-5-10 12:16:06   作者:   出處:
對外設進行 I/O 操作實際上也就是讀寫外設的寄存器,而我們通常使用的X86或者ARM處理器在硬體上決定了wince系統啟動後,無法直接存取物理地址,因此需要做一些工作來實現I/O操作. 

首先要理解 windows CE 下的地址映射機制。 wince有兩種地址:物理地址和虛擬位址.不同架構的 CPU 硬體上的區別導致地址映射也不同。MIPS和SH x 處理器,不採用MMU,直接在CPU和核心裡定義 1G 的物理地址;而X86和ARM帶有 MMU 單元,在 OEMAddressTable 中定義物理地址到虛擬位址間的映射關係或者是OS啟動後調用 CreateStaticMapping 和 NKCreateStaticMapping 來實現從虛擬位址到物理地址的靜態映射.經過靜態映射的地址,可以由作業系統核心用於 ISR 訪問裝置。如果我們要在應用程式中訪問外設,必須在物理地址和虛擬位址間建立動態映射關係,我們可以使用 VirtualAlloc 和 VirtualCopy (或者直接調用 MmmapIoSpace 函數)來實現。 

其次,如果是操作通過匯流排掛接的 I/O 或者儲存空間,必須先把匯流排地址轉化成 CPU 上的系統地址,再做物理地址到虛擬位址的映射。這裡需要查 CPU 的 Datasheet ,找出所要操作的I/O地址.先調用 HALTranslateBusAddress( )把匯流排地址轉化成CPU上的系統地址, 再調用 MmmapIoSpace 函數實現虛實映射;也可以使用 TransBusAddrToVirtual ()直接把匯流排上的地址轉化成系統的虛擬位址。 

第三,在一般的應用程式中訪問 I/O 是訪問它的緩衝段虛擬位址,而驅動中必須訪問無緩衝段虛擬位址。簡單來說無緩衝段虛擬位址 = 緩衝段虛擬位址 +0x20000000 。 

    總結起來,如果是 wince 核心(如HAL)訪問外部 I/O ,只需要在 OEMAddressTable 中定義物理地址到虛擬位址間的映射關係就可以了;如果是應用程式或者驅動要訪問 I/O ,要做的工作包括: 1 。在 CPU 物理地址和虛擬位址間做一個動態映射, 2 。對虛擬位址進行操作。

在X86和ARM架構的CPU中,wince訪問系統記憶體的方法隨程式所屬模式層次的不同而有所區別.
  1.在系統核心模式下(kernel mode),在OAL層訪問,只需要在OEMAddressTable 中做靜態虛真實位址映射就可以了.例如X86架構的映射表格式如下:
   ; OEMAddressTable defines the mapping between Physical and Virtual Address  // 定義4GB的虛擬位址和512MB儲存的映射關係
   ;   o MUST be in a READONLY Section
   ;   o First Entry MUST be RAM, mapping from 0x80000000 -> 0x00000000
   ;   o each entry is of the format ( VA, PA, cbSize )
   ;   o cbSize must be multiple of 4M
   ;   o last entry must be (0, 0, 0)
   ;   o must have at least one non-zero entry
   ; RAM 0x80000000 -> 0x00000000, size 64M       //把物理地址為0x00000000映射到虛擬位址為 0x80000000 處
   dd  80000000h,    0,   04000000h
   ; FLASH and other memory, if any
   ; dd  FlashVA,      FlashPA,    FlashSize
   ; Last entry, all zeros
   dd  0   0   0
2.在驅動或應用程式(user mode)中訪問RAM,既可以通過OEMAddressTable+VirtualCopy方式,也可以直接用MmMapIoSpace函數建立物理地址到當前進程虛擬位址的映射關係.
經過OEMAddressTable,實現的只是CPU物理地址到OS核心層虛擬位址的一次映射,如果需要在普通的應用程式中訪問記憶體,還要再用VirtuaAlloc+VirtualCopy做一個核心到當前進程的二次映射(有一種情況例外,就是你的OS被配置成Full Kernel Mode,這時任何應用程式都可以訪問OS核心地址).
     簡單說明幾個關鍵函數:
     VirtualAlloc用於在當前進程的虛擬位址空間中保留或者提交空間,在保留時以64KB為單位,提交時以4KB為單位。其函數原型為 

 LPVOID VirtualAlloc( 

  LPVOID lpAddress,  // 分配虛擬位址的起始指標 

  DWORD dwSize,     // 大小,以位元組為單位 

  DWORD flAllocationType, // 類型,設為MEM_RESERVE 

  DWORD flProtect    //  存取保護,設為PAGE_NOACCESS 

); 

  VirtualCopy 用來綁定物理地址到靜態映射虛擬位址: 

  BOOL VirtualCopy( 

  LPVOID lpvDest,         // 虛擬目的地址指標,接受VirtualAlloc的傳回值 

  LPVOID lpvSrc,         // 源物理地址指標 

  DWORD cbSize,          // 大小必須與虛擬位址相同 

  DWORD fdwProtect  // 存取保護類型 

); 

這裡需要注意的是 fdwProtect 參數。如果是驅動程式訪問,需要設定為 PAGE_NOCACHE ,以訪問無緩衝段虛擬位址。如果映射的物理位址範圍在 0x1FFFFFFF 之上,必須使用 PAGE_PHYSICAL ,此時必須把 lpvSrc 右移八位,實現地址對齊。(這是由核心中 VirtualCopy 的實現決定的,在那個函數中會判斷如果是 PAGE_PHYSICAL 就將 PHYSADDR 左移 8 位移回來,原始碼位於 private/winceos/coreos/nk/kernel 目錄下的 virtmem.c中的DoVirtualCopy ) 

      MmMapIoSpace 用來把物理地址直接映射到與進程無關的虛擬位址上。函數原型為 

 PVOID MmMapIoSpace( 

  PHYSICAL_ADDRESS PhysicalAddress, 

  ULONG NumberOfBytes, 

  BOOLEAN CacheEnable 

);

  一個使用 VirtualAlloc+Copy 的例子:把物理地址為 0x10000000 的單元映射到虛擬位址空間中。 

#include <windows.h> 

  

#define PHYSADDR  ((PVOID)0x10000000) 

// PHYSADDR is the physical address of the peripheral 

// registers 

  

#define SIZE  (4800*4) 

  

LPVOID lpv; 

BOOL bRet; 

  

lpv = VirtualAlloc(0, SIZE, MEM_RESERVE, PAGE_NOACCESS); 

// For a user mode driver, always leave the first 

// parameter 0 and use only the flags MEM_RESERVE 

// and PAGE_NOACCESS Check the return value: lpv == 0 

// is an error 

  

printf(TEXT("VirtualAlloc reservation @%8.8lx/r/n"), lpv); 

bRet = VirtualCopy(lpv, PHYSADDR>>8, SIZE, PAGE_READWRITE | PAGE_NOCACHE | PAGE_PHYSICAL); 

// The lpv parameter is the virtual address returned 

// by VirtualAlloc(). 

// Always use PAGE_NOCACHE */ 

  

// Check the return value: bRet ==0 is an error */ 

printf(TEXT("VirtualCopy returned: %d/r/n"), bRet); 

  

// At this point lpv is a virtual address which maps 

// the I/O registers 

// at PHYSADDR for SIZE bytes */ 

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.