uio_netx.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. /*
  2. * UIO driver for Hilscher NetX based fieldbus cards (cifX, comX).
  3. * See http://www.hilscher.com for details.
  4. *
  5. * (C) 2007 Hans J. Koch <hjk@linutronix.de>
  6. * (C) 2008 Manuel Traut <manut@linutronix.de>
  7. *
  8. * Licensed under GPL version 2 only.
  9. *
  10. */
  11. #include <linux/device.h>
  12. #include <linux/io.h>
  13. #include <linux/module.h>
  14. #include <linux/pci.h>
  15. #include <linux/uio_driver.h>
  16. #define PCI_VENDOR_ID_HILSCHER 0x15CF
  17. #define PCI_DEVICE_ID_HILSCHER_NETX 0x0000
  18. #define PCI_SUBDEVICE_ID_NXSB_PCA 0x3235
  19. #define PCI_SUBDEVICE_ID_NXPCA 0x3335
  20. #define DPM_HOST_INT_EN0 0xfff0
  21. #define DPM_HOST_INT_STAT0 0xffe0
  22. #define DPM_HOST_INT_MASK 0xe600ffff
  23. #define DPM_HOST_INT_GLOBAL_EN 0x80000000
  24. static irqreturn_t netx_handler(int irq, struct uio_info *dev_info)
  25. {
  26. void __iomem *int_enable_reg = dev_info->mem[0].internal_addr
  27. + DPM_HOST_INT_EN0;
  28. void __iomem *int_status_reg = dev_info->mem[0].internal_addr
  29. + DPM_HOST_INT_STAT0;
  30. /* Is one of our interrupts enabled and active ? */
  31. if (!(ioread32(int_enable_reg) & ioread32(int_status_reg)
  32. & DPM_HOST_INT_MASK))
  33. return IRQ_NONE;
  34. /* Disable interrupt */
  35. iowrite32(ioread32(int_enable_reg) & ~DPM_HOST_INT_GLOBAL_EN,
  36. int_enable_reg);
  37. return IRQ_HANDLED;
  38. }
  39. static int __devinit netx_pci_probe(struct pci_dev *dev,
  40. const struct pci_device_id *id)
  41. {
  42. struct uio_info *info;
  43. int bar;
  44. info = kzalloc(sizeof(struct uio_info), GFP_KERNEL);
  45. if (!info)
  46. return -ENOMEM;
  47. if (pci_enable_device(dev))
  48. goto out_free;
  49. if (pci_request_regions(dev, "netx"))
  50. goto out_disable;
  51. switch (id->device) {
  52. case PCI_DEVICE_ID_HILSCHER_NETX:
  53. bar = 0;
  54. info->name = "netx";
  55. break;
  56. default:
  57. bar = 2;
  58. info->name = "netx_plx";
  59. }
  60. /* BAR0 or 2 points to the card's dual port memory */
  61. info->mem[0].addr = pci_resource_start(dev, bar);
  62. if (!info->mem[0].addr)
  63. goto out_release;
  64. info->mem[0].internal_addr = ioremap(pci_resource_start(dev, bar),
  65. pci_resource_len(dev, bar));
  66. if (!info->mem[0].internal_addr)
  67. goto out_release;
  68. info->mem[0].size = pci_resource_len(dev, bar);
  69. info->mem[0].memtype = UIO_MEM_PHYS;
  70. info->irq = dev->irq;
  71. info->irq_flags = IRQF_SHARED;
  72. info->handler = netx_handler;
  73. info->version = "0.0.1";
  74. /* Make sure all interrupts are disabled */
  75. iowrite32(0, info->mem[0].internal_addr + DPM_HOST_INT_EN0);
  76. if (uio_register_device(&dev->dev, info))
  77. goto out_unmap;
  78. pci_set_drvdata(dev, info);
  79. dev_info(&dev->dev, "Found %s card, registered UIO device.\n",
  80. info->name);
  81. return 0;
  82. out_unmap:
  83. iounmap(info->mem[0].internal_addr);
  84. out_release:
  85. pci_release_regions(dev);
  86. out_disable:
  87. pci_disable_device(dev);
  88. out_free:
  89. kfree(info);
  90. return -ENODEV;
  91. }
  92. static void netx_pci_remove(struct pci_dev *dev)
  93. {
  94. struct uio_info *info = pci_get_drvdata(dev);
  95. /* Disable all interrupts */
  96. iowrite32(0, info->mem[0].internal_addr + DPM_HOST_INT_EN0);
  97. uio_unregister_device(info);
  98. pci_release_regions(dev);
  99. pci_disable_device(dev);
  100. pci_set_drvdata(dev, NULL);
  101. iounmap(info->mem[0].internal_addr);
  102. kfree(info);
  103. }
  104. static struct pci_device_id netx_pci_ids[] = {
  105. {
  106. .vendor = PCI_VENDOR_ID_HILSCHER,
  107. .device = PCI_DEVICE_ID_HILSCHER_NETX,
  108. .subvendor = 0,
  109. .subdevice = 0,
  110. },
  111. {
  112. .vendor = PCI_VENDOR_ID_PLX,
  113. .device = PCI_DEVICE_ID_PLX_9030,
  114. .subvendor = PCI_VENDOR_ID_PLX,
  115. .subdevice = PCI_SUBDEVICE_ID_NXSB_PCA,
  116. },
  117. {
  118. .vendor = PCI_VENDOR_ID_PLX,
  119. .device = PCI_DEVICE_ID_PLX_9030,
  120. .subvendor = PCI_VENDOR_ID_PLX,
  121. .subdevice = PCI_SUBDEVICE_ID_NXPCA,
  122. },
  123. { 0, }
  124. };
  125. static struct pci_driver netx_pci_driver = {
  126. .name = "netx",
  127. .id_table = netx_pci_ids,
  128. .probe = netx_pci_probe,
  129. .remove = netx_pci_remove,
  130. };
  131. static int __init netx_init_module(void)
  132. {
  133. return pci_register_driver(&netx_pci_driver);
  134. }
  135. static void __exit netx_exit_module(void)
  136. {
  137. pci_unregister_driver(&netx_pci_driver);
  138. }
  139. module_init(netx_init_module);
  140. module_exit(netx_exit_module);
  141. MODULE_DEVICE_TABLE(pci, netx_pci_ids);
  142. MODULE_LICENSE("GPL v2");
  143. MODULE_AUTHOR("Hans J. Koch, Manuel Traut");