pvpanic-pci.c 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Pvpanic PCI Device Support
  4. *
  5. * Copyright (C) 2021 Oracle.
  6. */
  7. #include <linux/kernel.h>
  8. #include <linux/module.h>
  9. #include <linux/pci.h>
  10. #include <linux/types.h>
  11. #include <linux/slab.h>
  12. #include <uapi/misc/pvpanic.h>
  13. #include "pvpanic.h"
  14. #define PCI_VENDOR_ID_REDHAT 0x1b36
  15. #define PCI_DEVICE_ID_REDHAT_PVPANIC 0x0011
  16. MODULE_AUTHOR("Mihai Carabas <[email protected]>");
  17. MODULE_DESCRIPTION("pvpanic device driver");
  18. MODULE_LICENSE("GPL");
  19. static ssize_t capability_show(struct device *dev, struct device_attribute *attr, char *buf)
  20. {
  21. struct pvpanic_instance *pi = dev_get_drvdata(dev);
  22. return sysfs_emit(buf, "%x\n", pi->capability);
  23. }
  24. static DEVICE_ATTR_RO(capability);
  25. static ssize_t events_show(struct device *dev, struct device_attribute *attr, char *buf)
  26. {
  27. struct pvpanic_instance *pi = dev_get_drvdata(dev);
  28. return sysfs_emit(buf, "%x\n", pi->events);
  29. }
  30. static ssize_t events_store(struct device *dev, struct device_attribute *attr,
  31. const char *buf, size_t count)
  32. {
  33. struct pvpanic_instance *pi = dev_get_drvdata(dev);
  34. unsigned int tmp;
  35. int err;
  36. err = kstrtouint(buf, 16, &tmp);
  37. if (err)
  38. return err;
  39. if ((tmp & pi->capability) != tmp)
  40. return -EINVAL;
  41. pi->events = tmp;
  42. return count;
  43. }
  44. static DEVICE_ATTR_RW(events);
  45. static struct attribute *pvpanic_pci_dev_attrs[] = {
  46. &dev_attr_capability.attr,
  47. &dev_attr_events.attr,
  48. NULL
  49. };
  50. ATTRIBUTE_GROUPS(pvpanic_pci_dev);
  51. static int pvpanic_pci_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
  52. {
  53. struct pvpanic_instance *pi;
  54. void __iomem *base;
  55. int ret;
  56. ret = pcim_enable_device(pdev);
  57. if (ret < 0)
  58. return ret;
  59. base = pcim_iomap(pdev, 0, 0);
  60. if (!base)
  61. return -ENOMEM;
  62. pi = devm_kmalloc(&pdev->dev, sizeof(*pi), GFP_KERNEL);
  63. if (!pi)
  64. return -ENOMEM;
  65. pi->base = base;
  66. pi->capability = PVPANIC_PANICKED | PVPANIC_CRASH_LOADED;
  67. /* initlize capability by RDPT */
  68. pi->capability &= ioread8(base);
  69. pi->events = pi->capability;
  70. return devm_pvpanic_probe(&pdev->dev, pi);
  71. }
  72. static const struct pci_device_id pvpanic_pci_id_tbl[] = {
  73. { PCI_DEVICE(PCI_VENDOR_ID_REDHAT, PCI_DEVICE_ID_REDHAT_PVPANIC)},
  74. {}
  75. };
  76. MODULE_DEVICE_TABLE(pci, pvpanic_pci_id_tbl);
  77. static struct pci_driver pvpanic_pci_driver = {
  78. .name = "pvpanic-pci",
  79. .id_table = pvpanic_pci_id_tbl,
  80. .probe = pvpanic_pci_probe,
  81. .driver = {
  82. .dev_groups = pvpanic_pci_dev_groups,
  83. },
  84. };
  85. module_pci_driver(pvpanic_pci_driver);