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

Linux/Unix编程

开发平台:

Unix_Linux

  1. /* 
  2.  *
  3.  * SNI64 specific PCI support for SNI IO.
  4.  *
  5.  * This file is subject to the terms and conditions of the GNU General Public
  6.  * License.  See the file "COPYING" in the main directory of this archive
  7.  * for more details.
  8.  *
  9.  * Copyright (c) 1997, 1998, 2000-2002 Silicon Graphics, Inc.  All rights reserved.
  10.  */
  11. #include <linux/init.h>
  12. #include <linux/types.h>
  13. #include <linux/config.h>
  14. #include <linux/pci.h>
  15. #include <asm/sn/types.h>
  16. #include <asm/sn/sgi.h>
  17. #include <asm/sn/io.h>
  18. #include <asm/sn/driver.h>
  19. #include <asm/sn/iograph.h>
  20. #include <asm/param.h>
  21. #include <asm/sn/pio.h>
  22. #include <asm/sn/xtalk/xwidget.h>
  23. #include <asm/sn/sn_private.h>
  24. #include <asm/sn/addrs.h>
  25. #include <asm/sn/invent.h>
  26. #include <asm/sn/hcl.h>
  27. #include <asm/sn/hcl_util.h>
  28. #include <asm/sn/pci/pciio.h>
  29. #include <asm/sn/pci/pcibr.h>
  30. #include <asm/sn/pci/pcibr_private.h>
  31. #include <asm/sn/pci/bridge.h>
  32. #ifdef DEBUG_CONFIG
  33. #define DBG(x...) printk(x)
  34. #else
  35. #define DBG(x...)
  36. #endif
  37. #ifdef CONFIG_PCI
  38. extern devfs_handle_t pci_bus_to_vertex(unsigned char);
  39. extern devfs_handle_t devfn_to_vertex(unsigned char bus, unsigned char devfn);
  40. /*
  41.  * snia64_read_config_byte - Read a byte from the config area of the device.
  42.  */
  43. static int snia64_read_config_byte (struct pci_dev *dev,
  44.                                    int where, unsigned char *val)
  45. {
  46. unsigned long res = 0;
  47. unsigned size = 1;
  48. devfs_handle_t device_vertex;
  49. if ( (dev == (struct pci_dev *)0) || (val == (unsigned char *)0) ) {
  50. return PCIBIOS_DEVICE_NOT_FOUND;
  51. }
  52. device_vertex = devfn_to_vertex(dev->bus->number, dev->devfn);
  53. if (!device_vertex) {
  54. DBG("%s : nonexistent device: bus= 0x%x  slot= 0x%x  func= 0x%xn", 
  55. __FUNCTION__, dev->bus->number, PCI_SLOT(dev->devfn), PCI_FUNC(dev->devfn));
  56. return(-1);
  57. }
  58. res = pciio_config_get(device_vertex, (unsigned) where, size);
  59. *val = (unsigned char) res;
  60. return PCIBIOS_SUCCESSFUL;
  61. }
  62. /*
  63.  * snia64_read_config_word - Read 2 bytes from the config area of the device.
  64.  */
  65. static int snia64_read_config_word (struct pci_dev *dev,
  66.                                    int where, unsigned short *val)
  67. {
  68. unsigned long res = 0;
  69. unsigned size = 2; /* 2 bytes */
  70. devfs_handle_t device_vertex;
  71. if ( (dev == (struct pci_dev *)0) || (val == (unsigned short *)0) ) {
  72. return PCIBIOS_DEVICE_NOT_FOUND;
  73. }
  74. device_vertex = devfn_to_vertex(dev->bus->number, dev->devfn);
  75. if (!device_vertex) {
  76. DBG("%s : nonexistent device: bus= 0x%x  slot= 0x%x  func= 0x%xn", 
  77. __FUNCTION__, dev->bus->number, PCI_SLOT(dev->devfn), PCI_FUNC(dev->devfn));
  78. return(-1);
  79. }
  80. res = pciio_config_get(device_vertex, (unsigned) where, size);
  81. *val = (unsigned short) res;
  82. return PCIBIOS_SUCCESSFUL;
  83. }
  84. /*
  85.  * snia64_read_config_dword - Read 4 bytes from the config area of the device.
  86.  */
  87. static int snia64_read_config_dword (struct pci_dev *dev,
  88.                                     int where, unsigned int *val)
  89. {
  90. unsigned long res = 0;
  91. unsigned size = 4; /* 4 bytes */
  92. devfs_handle_t device_vertex;
  93. if (where & 3) {
  94. return PCIBIOS_BAD_REGISTER_NUMBER;
  95. }
  96. if ( (dev == (struct pci_dev *)0) || (val == (unsigned int *)0) ) {
  97. return PCIBIOS_DEVICE_NOT_FOUND;
  98. }
  99. device_vertex = devfn_to_vertex(dev->bus->number, dev->devfn);
  100. if (!device_vertex) {
  101. DBG("%s : nonexistent device: bus= 0x%x  slot= 0x%x  func= 0x%xn", 
  102. __FUNCTION__, dev->bus->number, PCI_SLOT(dev->devfn), PCI_FUNC(dev->devfn));
  103. return(-1);
  104. }
  105. res = pciio_config_get(device_vertex, (unsigned) where, size);
  106. *val = (unsigned int) res;
  107. return PCIBIOS_SUCCESSFUL;
  108. }
  109. /*
  110.  * snia64_write_config_byte - Writes 1 byte to the config area of the device.
  111.  */
  112. static int snia64_write_config_byte (struct pci_dev *dev,
  113.                                     int where, unsigned char val)
  114. {
  115. devfs_handle_t device_vertex;
  116. if ( dev == (struct pci_dev *)0 ) {
  117. return PCIBIOS_DEVICE_NOT_FOUND;
  118. }
  119. /* 
  120.  * if it's an IOC3 then we bail out, we special
  121.  * case them with pci_fixup_ioc3
  122.  */
  123. if (dev->vendor == PCI_VENDOR_ID_SGI && 
  124.     dev->device == PCI_DEVICE_ID_SGI_IOC3 )
  125. return PCIBIOS_SUCCESSFUL;
  126. device_vertex = devfn_to_vertex(dev->bus->number, dev->devfn);
  127. if (!device_vertex) {
  128. DBG("%s : nonexistent device: bus= 0x%x  slot= 0x%x  func= 0x%xn", 
  129. __FUNCTION__, dev->bus->number, PCI_SLOT(dev->devfn), PCI_FUNC(dev->devfn));
  130. return(-1);
  131. }
  132. pciio_config_set( device_vertex, (unsigned)where, 1, (uint64_t) val);
  133. return PCIBIOS_SUCCESSFUL;
  134. }
  135. /*
  136.  * snia64_write_config_word - Writes 2 bytes to the config area of the device.
  137.  */
  138. static int snia64_write_config_word (struct pci_dev *dev,
  139.                                     int where, unsigned short val)
  140. {
  141. devfs_handle_t device_vertex = NULL;
  142. if (where & 1) {
  143. return PCIBIOS_BAD_REGISTER_NUMBER;
  144. }
  145. if ( dev == (struct pci_dev *)0 ) {
  146. return PCIBIOS_DEVICE_NOT_FOUND;
  147. }
  148. /* 
  149.  * if it's an IOC3 then we bail out, we special
  150.  * case them with pci_fixup_ioc3
  151.  */
  152. if (dev->vendor == PCI_VENDOR_ID_SGI && 
  153.     dev->device == PCI_DEVICE_ID_SGI_IOC3)
  154. return PCIBIOS_SUCCESSFUL;
  155. device_vertex = devfn_to_vertex(dev->bus->number, dev->devfn);
  156. if (!device_vertex) {
  157. DBG("%s : nonexistent device: bus= 0x%x  slot= 0x%x  func= 0x%xn", 
  158. __FUNCTION__, dev->bus->number, PCI_SLOT(dev->devfn), PCI_FUNC(dev->devfn));
  159. return(-1);
  160. }
  161. pciio_config_set( device_vertex, (unsigned)where, 2, (uint64_t) val);
  162. return PCIBIOS_SUCCESSFUL;
  163. }
  164. /*
  165.  * snia64_write_config_dword - Writes 4 bytes to the config area of the device.
  166.  */
  167. static int snia64_write_config_dword (struct pci_dev *dev,
  168.                                      int where, unsigned int val)
  169. {
  170. devfs_handle_t device_vertex;
  171. if (where & 3) {
  172. return PCIBIOS_BAD_REGISTER_NUMBER;
  173. }
  174. if ( dev == (struct pci_dev *)0 ) {
  175. return PCIBIOS_DEVICE_NOT_FOUND;
  176. }
  177. /* 
  178.  * if it's an IOC3 then we bail out, we special
  179.  * case them with pci_fixup_ioc3
  180.  */
  181. if (dev->vendor == PCI_VENDOR_ID_SGI && 
  182.     dev->device == PCI_DEVICE_ID_SGI_IOC3)
  183. return PCIBIOS_SUCCESSFUL;
  184. device_vertex = devfn_to_vertex(dev->bus->number, dev->devfn);
  185. if (!device_vertex) {
  186. DBG("%s : nonexistent device: bus= 0x%x  slot= 0x%x  func= 0x%xn", 
  187. __FUNCTION__, dev->bus->number, PCI_SLOT(dev->devfn), PCI_FUNC(dev->devfn));
  188. return(-1);
  189. }
  190. pciio_config_set( device_vertex, (unsigned)where, 4, (uint64_t) val);
  191. return PCIBIOS_SUCCESSFUL;
  192. }
  193. static struct pci_ops snia64_pci_ops = {
  194. snia64_read_config_byte,
  195. snia64_read_config_word,
  196. snia64_read_config_dword,
  197. snia64_write_config_byte,
  198. snia64_write_config_word,
  199. snia64_write_config_dword
  200. };
  201. /*
  202.  * snia64_pci_find_bios - SNIA64 pci_find_bios() platform specific code.
  203.  */
  204. void __init
  205. sn_pci_find_bios(void)
  206. {
  207. extern struct pci_ops pci_conf;
  208. /*
  209.  * Go initialize our IO Infrastructure ..
  210.  */
  211. extern void sgi_master_io_infr_init(void);
  212. sgi_master_io_infr_init();
  213. /* sn_io_infrastructure_init(); */
  214. pci_conf = snia64_pci_ops;
  215. }
  216. void
  217. pci_fixup_ioc3(struct pci_dev *d)
  218. {
  219.         int  i;
  220. unsigned int  size;
  221.         /* IOC3 only decodes 0x20 bytes of the config space, reading
  222.  * beyond that is relatively benign but writing beyond that
  223.  * (especially the base address registers) will shut down the
  224.  * pci bus...so avoid doing so.
  225.  * NOTE: this means we can't program the intr_pin into the device,
  226.  *       currently we hack this with special code in 
  227.  *  sgi_pci_intr_support()
  228.  */
  229.         DBG("pci_fixup_ioc3: Fixing base addresses for ioc3 device %sn", d->slot_name);
  230. /* I happen to know from the spec that the ioc3 needs only 0xfffff 
  231.  * The standard pci trick of writing ~0 to the baddr and seeing
  232.  * what comes back doesn't work with the ioc3
  233.  */
  234. size = 0xfffff;
  235. d->resource[0].end = (unsigned long) d->resource[0].start + (unsigned long) size;
  236. /*
  237.  * Zero out the resource structure .. because we did not go through 
  238.  * the normal PCI Infrastructure Init, garbbage are left in these 
  239.  * fileds.
  240.  */
  241.         for (i = 1; i <= PCI_ROM_RESOURCE; i++) {
  242.                 d->resource[i].start = 0UL;
  243.                 d->resource[i].end = 0UL;
  244.                 d->resource[i].flags = 0UL;
  245.         }
  246. /*
  247.  * Hardcode Device 4 register(IOC3 is in Slot 4) to set the 
  248.  * DEV_DIRECT bit.  This will not work if IOC3 is not on Slot 
  249.  * 4.
  250.  */
  251. DBG("pci_fixup_ioc3: FIXME .. need to take NASID into account when setting IOC3 devreg 0x%xn", *(volatile u32 *)0xc0000a000f000220);
  252.   *(volatile u32 *)0xc0000a000f000220 |= 0x90000; 
  253.         d->subsystem_vendor = 0;
  254.         d->subsystem_device = 0;
  255. }
  256. #else
  257. void sn_pci_find_bios(void) {}
  258. void pci_fixup_ioc3(struct pci_dev *d) {}
  259. struct list_head pci_root_buses;
  260. struct list_head pci_root_buses;
  261. struct list_head pci_devices;
  262. #endif /* CONFIG_PCI */