eeh.h
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:6k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /* 
  2.  * eeh.h
  3.  * Copyright (C) 2001  Dave Engebretsen & Todd Inglett IBM Corporation.
  4.  *
  5.  * This program is free software; you can redistribute it and/or modify
  6.  * it under the terms of the GNU General Public License as published by
  7.  * the Free Software Foundation; either version 2 of the License, or
  8.  * (at your option) any later version.
  9.  * 
  10.  * This program is distributed in the hope that it will be useful,
  11.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  13.  * GNU General Public License for more details.
  14.  * 
  15.  * You should have received a copy of the GNU General Public License
  16.  * along with this program; if not, write to the Free Software
  17.  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
  18.  */
  19. /* Start Change Log
  20.  * 2001/10/27 : engebret : Created.
  21.  * End Change Log 
  22.  */
  23. #ifndef _EEH_H
  24. #define _EEH_H
  25. struct pci_dev;
  26. #define IO_UNMAPPED_REGION_ID 0xaUL
  27. #define IO_TOKEN_TO_ADDR(token) ((((unsigned long)(token)) & 0xFFFFFFFF) | (0xEUL << 60))
  28. /* Flag bits encoded in the 3 unused function bits of devfn */
  29. #define EEH_TOKEN_DISABLED (1UL << 34UL) /* eeh is disabled for this token */
  30. #define IS_EEH_TOKEN_DISABLED(token) ((unsigned long)(token) & EEH_TOKEN_DISABLED)
  31. #define EEH_STATE_OVERRIDE 1   /* IOA does not require eeh traps */
  32. #define EEH_STATE_FAILURE  16  /* */
  33. /* This is for profiling only */
  34. extern unsigned long eeh_total_mmio_ffs;
  35. extern int eeh_implemented;
  36. void eeh_init(void);
  37. static inline int is_eeh_implemented(void) { return eeh_implemented; }
  38. int eeh_get_state(unsigned long ea);
  39. unsigned long eeh_check_failure(void *token, unsigned long val);
  40. #define EEH_DISABLE 0
  41. #define EEH_ENABLE 1
  42. #define EEH_RELEASE_LOADSTORE 2
  43. #define EEH_RELEASE_DMA 3
  44. int eeh_set_option(struct pci_dev *dev, int options);
  45. /* Given a PCI device check if eeh should be configured or not.
  46.  * This may look at firmware properties and/or kernel cmdline options.
  47.  */
  48. int is_eeh_configured(struct pci_dev *dev);
  49. /* Generate an EEH token.
  50.  * The high nibble of the offset is cleared, otherwise bounds checking is performed.
  51.  * Use IO_TOKEN_TO_ADDR(token) to translate this token back to a mapped virtual addr.
  52.  * Do NOT do this to perform IO -- use the read/write macros!
  53.  */
  54. unsigned long eeh_token(unsigned long phb,
  55. unsigned long bus,
  56. unsigned long devfn,
  57. unsigned long offset);
  58. extern void *memcpy(void *, const void *, unsigned long);
  59. extern void *memset(void *,int, unsigned long);
  60. /* EEH_POSSIBLE_ERROR() -- test for possible MMIO failure.
  61.  *
  62.  * Order this macro for performance.
  63.  * If EEH is off for a device and it is a memory BAR, ioremap will
  64.  * map it to the IOREGION.  In this case addr == vaddr and since these
  65.  * should be in registers we compare them first.  Next we check for
  66.  * all ones which is perhaps fastest as ~val.  Finally we weed out
  67.  * EEH disabled IO BARs.
  68.  *
  69.  * If this macro yields TRUE, the caller relays to eeh_check_failure()
  70.  * which does further tests out of line.
  71.  */
  72. /* #define EEH_POSSIBLE_ERROR(addr, vaddr, val) ((vaddr) != (addr) && ~(val) == 0 && !IS_EEH_TOKEN_DISABLED(addr)) */
  73. /* This version is rearranged to collect some profiling data */
  74. #define EEH_POSSIBLE_ERROR(addr, vaddr, val) (~(val) == 0 && (++eeh_total_mmio_ffs, (vaddr) != (addr) && !IS_EEH_TOKEN_DISABLED(addr)))
  75. /* 
  76.  * MMIO read/write operations with EEH support.
  77.  *
  78.  * addr: 64b token of the form 0xA0PPBBDDyyyyyyyy
  79.  *       0xA0     : Unmapped MMIO region
  80.  *       PP       : PHB index (starting at zero)
  81.  *  BB   : PCI Bus number under given PHB
  82.  *  DD   : PCI devfn under given bus
  83.  *       yyyyyyyy : Virtual address offset
  84.  * 
  85.  * An actual virtual address is produced from this token
  86.  * by masking into the form:
  87.  *   0xE0000000yyyyyyyy
  88.  */
  89. static inline u8 eeh_readb(void *addr) {
  90. volatile u8 *vaddr = (volatile u8 *)IO_TOKEN_TO_ADDR(addr);
  91. u8 val = in_8(vaddr);
  92. if (EEH_POSSIBLE_ERROR(addr, vaddr, val))
  93. return eeh_check_failure(addr, val);
  94. return val;
  95. }
  96. static inline void eeh_writeb(u8 val, void *addr) {
  97. volatile u8 *vaddr = (volatile u8 *)IO_TOKEN_TO_ADDR(addr);
  98. out_8(vaddr, val);
  99. }
  100. static inline u16 eeh_readw(void *addr) {
  101. volatile u16 *vaddr = (volatile u16 *)IO_TOKEN_TO_ADDR(addr);
  102. u16 val = in_le16(vaddr);
  103. if (EEH_POSSIBLE_ERROR(addr, vaddr, val))
  104. return eeh_check_failure(addr, val);
  105. return val;
  106. }
  107. static inline void eeh_writew(u16 val, void *addr) {
  108. volatile u16 *vaddr = (volatile u16 *)IO_TOKEN_TO_ADDR(addr);
  109. out_le16(vaddr, val);
  110. }
  111. static inline u32 eeh_readl(void *addr) {
  112. volatile u32 *vaddr = (volatile u32 *)IO_TOKEN_TO_ADDR(addr);
  113. u32 val = in_le32(vaddr);
  114. if (EEH_POSSIBLE_ERROR(addr, vaddr, val))
  115. return eeh_check_failure(addr, val);
  116. return val;
  117. }
  118. static inline void eeh_writel(u32 val, void *addr) {
  119. volatile u32 *vaddr = (volatile u32 *)IO_TOKEN_TO_ADDR(addr);
  120. out_le32(vaddr, val);
  121. }
  122. static inline void eeh_memset_io(void *addr, int c, unsigned long n) {
  123. void *vaddr = (void *)IO_TOKEN_TO_ADDR(addr);
  124. memset(vaddr, c, n);
  125. }
  126. static inline void eeh_memcpy_fromio(void *dest, void *src, unsigned long n) {
  127. void *vsrc = (void *)IO_TOKEN_TO_ADDR(src);
  128. memcpy(dest, vsrc, n);
  129. /* look for ffff's here at dest[n] */
  130. }
  131. static inline void eeh_memcpy_toio(void *dest, void *src, unsigned long n) {
  132. void *vdest = (void *)IO_TOKEN_TO_ADDR(dest);
  133. memcpy(vdest, src, n);
  134. }
  135. static inline void eeh_insb(volatile u8 *addr, void *buf, int n) {
  136. volatile u8 *vaddr = (volatile u8 *)IO_TOKEN_TO_ADDR(addr);
  137. _insb(vaddr, buf, n);
  138. /* ToDo: look for ff's in buf[n] */
  139. }
  140. static inline void eeh_outsb(volatile u8 *addr, const void *buf, int n) {
  141. volatile u8 *vaddr = (volatile u8 *)IO_TOKEN_TO_ADDR(addr);
  142. _outsb(vaddr, buf, n);
  143. }
  144. static inline void eeh_insw_ns(volatile u16 *addr, void *buf, int n) {
  145. volatile u16 *vaddr = (volatile u16 *)IO_TOKEN_TO_ADDR(addr);
  146. _insw_ns(vaddr, buf, n);
  147. /* ToDo: look for ffff's in buf[n] */
  148. }
  149. static inline void eeh_outsw_ns(volatile u16 *addr, const void *buf, int n) {
  150. volatile u16 *vaddr = (volatile u16 *)IO_TOKEN_TO_ADDR(addr);
  151. _outsw_ns(vaddr, buf, n);
  152. }
  153. static inline void eeh_insl_ns(volatile u32 *addr, void *buf, int n) {
  154. volatile u32 *vaddr = (volatile u32 *)IO_TOKEN_TO_ADDR(addr);
  155. _insl_ns(vaddr, buf, n);
  156. /* ToDo: look for ffffffff's in buf[n] */
  157. }
  158. static inline void eeh_outsl_ns(volatile u32 *addr, const void *buf, int n) {
  159. volatile u32 *vaddr = (volatile u32 *)IO_TOKEN_TO_ADDR(addr);
  160. _outsl_ns(vaddr, buf, n);
  161. }
  162. #endif /* _EEH_H */