plf_io.h
来自「eCos操作系统源码」· C头文件 代码 · 共 252 行
H
252 行
#ifndef CYGONCE_PLF_IO_H#define CYGONCE_PLF_IO_H//=============================================================================//// plf_io.h//// Platform specific IO support////=============================================================================//####ECOSGPLCOPYRIGHTBEGIN####// -------------------------------------------// This file is part of eCos, the Embedded Configurable Operating System.// Copyright (C) 1998, 1999, 2000, 2001, 2002 Red Hat, Inc.//// eCos is free software; you can redistribute it and/or modify it under// the terms of the GNU General Public License as published by the Free// Software Foundation; either version 2 or (at your option) any later version.//// eCos is distributed in the hope that it will be useful, but WITHOUT ANY// WARRANTY; without even the implied warranty of MERCHANTABILITY or// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License// for more details.//// You should have received a copy of the GNU General Public License along// with eCos; if not, write to the Free Software Foundation, Inc.,// 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.//// As a special exception, if other files instantiate templates or use macros// or inline functions from this file, or you compile this file and link it// with other works to produce a work based on this file, this file does not// by itself cause the resulting work to be covered by the GNU General Public// License. However the source code for this file must still be made available// in accordance with section (3) of the GNU General Public License.//// This exception does not invalidate any other reasons why a work based on// this file might be covered by the GNU General Public License.//// Alternative licenses for eCos may be arranged by contacting Red Hat, Inc.// at http://sources.redhat.com/ecos/ecos-license/// -------------------------------------------//####ECOSGPLCOPYRIGHTEND####//=============================================================================//#####DESCRIPTIONBEGIN####//// Author(s): Philippe Robin// Contributors: David A Rusling// Date: November 7, 2000// Purpose: Integrator PCI IO support macros// Description: // Usage: #include <cyg/hal/plf_io.h>////####DESCRIPTIONEND####////=============================================================================#include <pkgconf/hal_arm_integrator.h>#include <cyg/hal/hal_integrator.h>#include <cyg/hal/hal_platform_ints.h> // Interrupt vectors__externC void cyg_plf_pci_init(void);// Initialization of the PCI bus.#define HAL_PCI_INIT() cyg_plf_pci_init()// V3 access routines#define _V3Write16(o,v) (*(volatile cyg_uint16 *)(PCI_V3_BASE + (cyg_uint32)(o)) \ = (cyg_uint16)(v))#define _V3Read16(o) (*(volatile cyg_uint16 *)(PCI_V3_BASE + (cyg_uint32)(o)))#define _V3Write32(o,v) (*(volatile cyg_uint32 *)(PCI_V3_BASE + (cyg_uint32)(o)) \ = (cyg_uint32)(v))#define _V3Read32(o) (*(volatile cyg_uint32 *)(PCI_V3_BASE + (cyg_uint32)(o)))// _V3OpenConfigWindow - open V3 configuration window#define _V3OpenConfigWindow() \ { \ /* Set up base0 to see all 512Mbytes of memory space (not */ \ /* prefetchable), this frees up base1 for re-use by configuration*/ \ /* memory */ \ \ _V3Write32 (V3_LB_BASE0, ((INTEGRATOR_PCI_BASE & 0xFFF00000) | \ 0x90 | V3_LB_BASE_M_ENABLE)); \ /* Set up base1 to point into configuration space, note that MAP1 */ \ /* register is set up by pciMakeConfigAddress(). */ \ \ _V3Write32 (V3_LB_BASE1, ((CPU_PCI_CNFG_ADRS & 0xFFF00000) | \ 0x40 | V3_LB_BASE_M_ENABLE)); \ }// _V3CloseConfigWindow - close V3 configuration window#define _V3CloseConfigWindow() \ { \ /* Reassign base1 for use by prefetchable PCI memory */ \ _V3Write32 (V3_LB_BASE1, (((INTEGRATOR_PCI_BASE + SZ_256M) & 0xFFF00000) \ | 0x84 | V3_LB_BASE_M_ENABLE)); \ _V3Write16 (V3_LB_MAP1, \ (((INTEGRATOR_PCI_BASE + SZ_256M) & 0xFFF00000) >> 16) | 0x0006); \ \ /* And shrink base0 back to a 256M window (NOTE: MAP0 already correct) */ \ \ _V3Write32 (V3_LB_BASE0, ((INTEGRATOR_PCI_BASE & 0xFFF00000) | \ 0x80 | V3_LB_BASE_M_ENABLE)); \ }// Compute address necessary to access PCI config space for the given// bus and device.#define HAL_PCI_CONFIG_ADDRESS( __bus, __devfn, __offset ) \ ({ \ cyg_uint32 __address, __devicebit; \ cyg_uint16 __mapaddress; \ cyg_uint32 __dev = CYG_PCI_DEV_GET_DEV(__devfn); /* FIXME to check!! (slot?) */ \ \ if (__bus == 0) { \ /* local bus segment so need a type 0 config cycle */ \ /* build the PCI configuration "address" with one-hot in A31-A11 */ \ __address = PCI_CONFIG_BASE; \ __address |= ((__devfn & 0x07) << 8); \ __address |= __offset & 0xFF; \ __mapaddress = 0x000A; /* 101=>config cycle, 0=>A1=A0=0 */ \ __devicebit = (1 << (__dev + 11)); \ \ if ((__devicebit & 0xFF000000) != 0) { \ /* high order bits are handled by the MAP register */ \ __mapaddress |= (__devicebit >> 16); \ } else { \ /* low order bits handled directly in the address */ \ __address |= __devicebit; \ } \ } else { /* bus !=0 */ \ /* not the local bus segment so need a type 1 config cycle */ \ /* A31-A24 are don't care (so clear to 0) */ \ __mapaddress = 0x000B; /* 101=>config cycle, 1=>A1&A0 from PCI_CFG */ \ __address = PCI_CONFIG_BASE; \ __address |= ((__bus & 0xFF) << 16); /* bits 23..16 = bus number */ \ __address |= ((__dev & 0x1F) << 11); /* bits 15..11 = device number */ \ __address |= ((__devfn & 0x07) << 8); /* bits 10..8 = function number*/ \ __address |= __offset & 0xFF; /* bits 7..0 = register number*/ \ } \ _V3Write16(V3_LB_MAP1, __mapaddress); \ \ __address; \ })// Read a value from the PCI configuration space of the appropriate// size at an address composed from the bus, devfn and offset.#define HAL_PCI_CFG_READ_UINT8( __bus, __devfn, __offset, __val ) \ { \ _V3OpenConfigWindow(); \ __val = *(volatile cyg_uint8 *)HAL_PCI_CONFIG_ADDRESS(__bus, __devfn, __offset); \ _V3CloseConfigWindow(); \ }#define HAL_PCI_CFG_READ_UINT16( __bus, __devfn, __offset, __val ) \ { \ _V3OpenConfigWindow(); \ __val = *(volatile cyg_uint16 *)HAL_PCI_CONFIG_ADDRESS(__bus, __devfn, __offset); \ _V3CloseConfigWindow(); \ }#define HAL_PCI_CFG_READ_UINT32( __bus, __devfn, __offset, __val ) \ { \ _V3OpenConfigWindow(); \ __val = *(volatile cyg_uint16 *)HAL_PCI_CONFIG_ADDRESS(__bus, __devfn, __offset); \ __val |= (*(volatile cyg_uint16 *)HAL_PCI_CONFIG_ADDRESS(__bus, __devfn, (__offset+2)))<<16; \ _V3CloseConfigWindow(); \ }// Write a value to the PCI configuration space of the appropriate// size at an address composed from the bus, devfn and offset.#define HAL_PCI_CFG_WRITE_UINT8( __bus, __devfn, __offset, __val ) \ { \ _V3OpenConfigWindow(); \ *(volatile cyg_uint8 *)HAL_PCI_CONFIG_ADDRESS(__bus, __devfn, __offset) = __val; \ _V3CloseConfigWindow(); \ }#define HAL_PCI_CFG_WRITE_UINT16( __bus, __devfn, __offset, __val ) \ { \ _V3OpenConfigWindow(); \ *(volatile cyg_uint16 *)HAL_PCI_CONFIG_ADDRESS(__bus, __devfn, __offset) = __val; \ _V3CloseConfigWindow(); \ }#define HAL_PCI_CFG_WRITE_UINT32( __bus, __devfn, __offset, __val ) \ { \ _V3OpenConfigWindow(); \ *(volatile cyg_uint16 *)HAL_PCI_CONFIG_ADDRESS((__bus), (__devfn), (__offset)) = ((__val) & 0xFFFF); \ *(volatile cyg_uint16 *)HAL_PCI_CONFIG_ADDRESS((__bus), (__devfn), ((__offset)+2)) = (((__val)>>16) & 0xFFFF); \ _V3CloseConfigWindow(); \ }//-----------------------------------------------------------------------------// Resources// Map PCI device resources starting from these addresses in PCI space.#define HAL_PCI_ALLOC_BASE_MEMORY 0#define HAL_PCI_ALLOC_BASE_IO 0x4000// This is where the PCI spaces are mapped in the CPU's address space.#define HAL_PCI_PHYSICAL_MEMORY_BASE (PCI_MEM_BASE)#define HAL_PCI_PHYSICAL_IO_BASE (PCI_IO_BASE)// Translate the PCI interrupt requested by the device (INTA#, INTB#,// INTC# or INTD#) to the associated CPU interrupt (i.e., HAL vector).#define INTA CYGNUM_HAL_INTERRUPT_PCIINT0#define INTB CYGNUM_HAL_INTERRUPT_PCIINT1#define INTC CYGNUM_HAL_INTERRUPT_PCIINT2#define INTD CYGNUM_HAL_INTERRUPT_PCIINT3#define HAL_PCI_TRANSLATE_INTERRUPT( __bus, __devfn, __vec, __valid) \ CYG_MACRO_START \ cyg_uint8 __req; \ cyg_uint32 __dev; \ /* DANGER! For now this is the SDM interrupt table... */ \ static const cyg_uint8 irq_tab[12][4] = { \ /* INTA INTB INTC INTD */ \ {INTA, INTB, INTC, INTD}, /* idsel 20, slot 9 */ \ {INTB, INTC, INTD, INTA}, /* idsel 21, slot 10 */ \ {INTC, INTD, INTA, INTB}, /* idsel 22, slot 11 */ \ {INTD, INTA, INTB, INTC}, /* idsel 23, slot 12 */ \ {INTA, INTB, INTC, INTD}, /* idsel 24, slot 13 */ \ {INTB, INTC, INTD, INTA}, /* idsel 25, slot 14 */ \ {INTC, INTD, INTA, INTB}, /* idsel 26, slot 15 */ \ {INTD, INTA, INTB, INTC}, /* idsel 27, slot 16 */ \ {INTA, INTB, INTC, INTD}, /* idsel 28, slot 17 */ \ {INTB, INTC, INTD, INTA}, /* idsel 29, slot 18 */ \ {INTC, INTD, INTA, INTB}, /* idsel 30, slot 19 */ \ {INTD, INTA, INTB, INTC} /* idsel 31, slot 20 */ \ }; \ HAL_PCI_CFG_READ_UINT8(__bus, __devfn, CYG_PCI_CFG_INT_PIN, __req); \ __dev = CYG_PCI_DEV_GET_DEV(__devfn); /* FIXME to check!! (slot?)*/ \ \ /* if PIN = 0, default to A */ \ if (__req == 0) \ __req = 1; \ \ __vec = irq_tab[__dev - 9][__req - 1]; \ __valid = true; \ CYG_MACRO_END//-----------------------------------------------------------------------------#define CYGARC_PHYSICAL_ADDRESS(x) (x)//-----------------------------------------------------------------------------// end of plf_io.h#endif // CYGONCE_PLF_IO_H
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?