pci-stub.c 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Simple stub driver to reserve a PCI device
  4. *
  5. * Copyright (C) 2008 Red Hat, Inc.
  6. * Author:
  7. * Chris Wright
  8. *
  9. * Usage is simple, allocate a new id to the stub driver and bind the
  10. * device to it. For example:
  11. *
  12. * # echo "8086 10f5" > /sys/bus/pci/drivers/pci-stub/new_id
  13. * # echo -n 0000:00:19.0 > /sys/bus/pci/drivers/e1000e/unbind
  14. * # echo -n 0000:00:19.0 > /sys/bus/pci/drivers/pci-stub/bind
  15. * # ls -l /sys/bus/pci/devices/0000:00:19.0/driver
  16. * .../0000:00:19.0/driver -> ../../../bus/pci/drivers/pci-stub
  17. */
  18. #include <linux/module.h>
  19. #include <linux/pci.h>
  20. static char ids[1024] __initdata;
  21. module_param_string(ids, ids, sizeof(ids), 0);
  22. MODULE_PARM_DESC(ids, "Initial PCI IDs to add to the stub driver, format is "
  23. "\"vendor:device[:subvendor[:subdevice[:class[:class_mask]]]]\""
  24. " and multiple comma separated entries can be specified");
  25. static int pci_stub_probe(struct pci_dev *dev, const struct pci_device_id *id)
  26. {
  27. pci_info(dev, "claimed by stub\n");
  28. return 0;
  29. }
  30. static struct pci_driver stub_driver = {
  31. .name = "pci-stub",
  32. .id_table = NULL, /* only dynamic id's */
  33. .probe = pci_stub_probe,
  34. .driver_managed_dma = true,
  35. };
  36. static int __init pci_stub_init(void)
  37. {
  38. char *p, *id;
  39. int rc;
  40. rc = pci_register_driver(&stub_driver);
  41. if (rc)
  42. return rc;
  43. /* no ids passed actually */
  44. if (ids[0] == '\0')
  45. return 0;
  46. /* add ids specified in the module parameter */
  47. p = ids;
  48. while ((id = strsep(&p, ","))) {
  49. unsigned int vendor, device, subvendor = PCI_ANY_ID,
  50. subdevice = PCI_ANY_ID, class = 0, class_mask = 0;
  51. int fields;
  52. if (!strlen(id))
  53. continue;
  54. fields = sscanf(id, "%x:%x:%x:%x:%x:%x",
  55. &vendor, &device, &subvendor, &subdevice,
  56. &class, &class_mask);
  57. if (fields < 2) {
  58. pr_warn("pci-stub: invalid ID string \"%s\"\n", id);
  59. continue;
  60. }
  61. pr_info("pci-stub: add %04X:%04X sub=%04X:%04X cls=%08X/%08X\n",
  62. vendor, device, subvendor, subdevice, class, class_mask);
  63. rc = pci_add_dynid(&stub_driver, vendor, device,
  64. subvendor, subdevice, class, class_mask, 0);
  65. if (rc)
  66. pr_warn("pci-stub: failed to add dynamic ID (%d)\n",
  67. rc);
  68. }
  69. return 0;
  70. }
  71. static void __exit pci_stub_exit(void)
  72. {
  73. pci_unregister_driver(&stub_driver);
  74. }
  75. module_init(pci_stub_init);
  76. module_exit(pci_stub_exit);
  77. MODULE_LICENSE("GPL");
  78. MODULE_AUTHOR("Chris Wright <[email protected]>");