module.c 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * module.c - module sysfs fun for drivers
  4. */
  5. #include <linux/device.h>
  6. #include <linux/module.h>
  7. #include <linux/errno.h>
  8. #include <linux/slab.h>
  9. #include <linux/string.h>
  10. #include "base.h"
  11. static char *make_driver_name(const struct device_driver *drv)
  12. {
  13. char *driver_name;
  14. driver_name = kasprintf(GFP_KERNEL, "%s:%s", drv->bus->name, drv->name);
  15. if (!driver_name)
  16. return NULL;
  17. return driver_name;
  18. }
  19. static void module_create_drivers_dir(struct module_kobject *mk)
  20. {
  21. static DEFINE_MUTEX(drivers_dir_mutex);
  22. mutex_lock(&drivers_dir_mutex);
  23. if (mk && !mk->drivers_dir)
  24. mk->drivers_dir = kobject_create_and_add("drivers", &mk->kobj);
  25. mutex_unlock(&drivers_dir_mutex);
  26. }
  27. int module_add_driver(struct module *mod, const struct device_driver *drv)
  28. {
  29. char *driver_name;
  30. struct module_kobject *mk = NULL;
  31. int ret;
  32. if (!drv)
  33. return 0;
  34. if (mod)
  35. mk = &mod->mkobj;
  36. else if (drv->mod_name) {
  37. /* Lookup or create built-in module entry in /sys/modules */
  38. mk = lookup_or_create_module_kobject(drv->mod_name);
  39. if (mk) {
  40. /* remember our module structure */
  41. drv->p->mkobj = mk;
  42. /* lookup_or_create_module_kobject took a reference */
  43. kobject_put(&mk->kobj);
  44. }
  45. }
  46. if (!mk)
  47. return 0;
  48. ret = sysfs_create_link(&drv->p->kobj, &mk->kobj, "module");
  49. if (ret)
  50. return ret;
  51. driver_name = make_driver_name(drv);
  52. if (!driver_name) {
  53. ret = -ENOMEM;
  54. goto out_remove_kobj;
  55. }
  56. module_create_drivers_dir(mk);
  57. if (!mk->drivers_dir) {
  58. ret = -EINVAL;
  59. goto out_free_driver_name;
  60. }
  61. ret = sysfs_create_link(mk->drivers_dir, &drv->p->kobj, driver_name);
  62. if (ret)
  63. goto out_remove_drivers_dir;
  64. kfree(driver_name);
  65. return 0;
  66. out_remove_drivers_dir:
  67. sysfs_remove_link(mk->drivers_dir, driver_name);
  68. out_free_driver_name:
  69. kfree(driver_name);
  70. out_remove_kobj:
  71. sysfs_remove_link(&drv->p->kobj, "module");
  72. return ret;
  73. }
  74. void module_remove_driver(const struct device_driver *drv)
  75. {
  76. struct module_kobject *mk = NULL;
  77. char *driver_name;
  78. if (!drv)
  79. return;
  80. sysfs_remove_link(&drv->p->kobj, "module");
  81. if (drv->owner)
  82. mk = &drv->owner->mkobj;
  83. else if (drv->p->mkobj)
  84. mk = drv->p->mkobj;
  85. if (mk && mk->drivers_dir) {
  86. driver_name = make_driver_name(drv);
  87. if (driver_name) {
  88. sysfs_remove_link(mk->drivers_dir, driver_name);
  89. kfree(driver_name);
  90. }
  91. }
  92. }