mod.c 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. #include <linux/init.h>
  2. #include <linux/module.h>
  3. #include <linux/device.h>
  4. #include <linux/types.h>
  5. #include <linux/cdev.h>
  6. #include <linux/fs.h>
  7. #include <linux/slab.h>
  8. #include "mod.h"
  9. #include "dev.h"
  10. #include "debug.h"
  11. MODULE_LICENSE("GPL v2");
  12. MODULE_AUTHOR("Suren A. Chilingaryan <csa@suren.me>");
  13. MODULE_DESCRIPTION("Kernel Memory Manager - a module to manipulate kernel memory");
  14. MODULE_VERSION("0.0.1");
  15. int kmm_minors = KMM_MINORS;
  16. module_param(kmm_minors, int, S_IRUGO);
  17. static dev_t kmm_devno; /**< major number */
  18. static kmm_dev_t **kmm_devs = NULL; /**< per-device context */
  19. static spinlock_t kmm_devs_lock; /**< lock protecting creation/destruction of devices */
  20. static struct class *kmm_class; /**< device class */
  21. static void kmm_module_cleanup(void)
  22. {
  23. int i;
  24. if (kmm_devs) {
  25. for (i = 0; i < kmm_minors; i++) {
  26. if (kmm_devs[i]) {
  27. if (kmm_devs[i]->dev)
  28. device_destroy(kmm_class, kmm_devs[i]->devno);
  29. cdev_del(&kmm_devs[i]->cdev);
  30. kfree(kmm_devs[i]);
  31. }
  32. }
  33. kfree(kmm_devs);
  34. }
  35. if (kmm_class)
  36. class_destroy(kmm_class);
  37. unregister_chrdev_region(kmm_devno, kmm_minors);
  38. }
  39. static int kmm_module_setup_cdev(void)
  40. {
  41. int i;
  42. int err = 0;
  43. dev_t devno;
  44. kmm_dev_t *dev;
  45. dev = kmalloc(sizeof(kmm_dev_t), GFP_KERNEL);
  46. if (!dev) {
  47. mod_info("Couldn't allocate memory. Device is not created.\n");
  48. return -ENOMEM;
  49. }
  50. cdev_init(&dev->cdev, kmm_get_fops());
  51. dev->cdev.owner = THIS_MODULE;
  52. dev->cdev.ops = kmm_get_fops();
  53. spin_lock(&kmm_devs_lock);
  54. for (i = 0; i < kmm_minors; i++) {
  55. if (!kmm_devs[i])
  56. break;
  57. }
  58. if (i == kmm_minors) {
  59. mod_info("No free minor numbers left");
  60. err = -EBUSY;
  61. goto fail;
  62. }
  63. devno = MKDEV(MAJOR(kmm_devno), MINOR(kmm_devno) + i);
  64. dev->devno = devno;
  65. err = cdev_add(&dev->cdev, devno, 1);
  66. if (err) {
  67. mod_info("Error %d adding device kmm%d", err, i);
  68. goto fail;
  69. }
  70. dev->dev = device_create(kmm_class, NULL, devno, dev, KMM_NODE_FMT, MINOR(kmm_devno) + i);
  71. if (!dev->dev) {
  72. mod_info("Error creating /dev/%s%d\n", KMM_NODE, MINOR(kmm_devno) + i);
  73. goto fail;
  74. }
  75. dev_set_drvdata(dev->dev, dev);
  76. kmm_devs[i] = dev;
  77. spin_unlock(&kmm_devs_lock);
  78. mod_info("Device /dev/%s%d added\n", KMM_NODE, MINOR(kmm_devno) + i);
  79. return 0;
  80. fail:
  81. spin_unlock(&kmm_devs_lock);
  82. kfree(dev);
  83. return err;
  84. }
  85. static int __init kmm_module_init(void)
  86. {
  87. int err;
  88. spin_lock_init(&kmm_devs_lock);
  89. if ((err = alloc_chrdev_region(&kmm_devno, 0, kmm_minors, KMM_NODE))) {
  90. mod_info("Couldn't allocate chrdev region. Module not loaded.\n");
  91. goto alloc_chrdev_fail;
  92. }
  93. kmm_class = class_create(THIS_MODULE, KMM_NODE);
  94. if (IS_ERR(kmm_class)) {
  95. mod_info("No sysfs support. Module not loaded.\n");
  96. goto fail;
  97. }
  98. kmm_devs = kmalloc(kmm_minors * sizeof(kmm_dev_t*), GFP_KERNEL);
  99. if (!kmm_devs) {
  100. mod_info("Couldn't allocate memory. Module not loaded.\n");
  101. err = -ENOMEM;
  102. }
  103. memset(kmm_devs, 0, kmm_minors * sizeof(kmm_dev_t*));
  104. mod_info("Major %d allocated to node '%s'\n", MAJOR(kmm_devno), KMM_NODE);
  105. err = kmm_module_setup_cdev();
  106. if (err) goto fail;
  107. return 0;
  108. fail:
  109. kmm_module_cleanup();
  110. alloc_chrdev_fail:
  111. return err;
  112. }
  113. static void __exit kmm_module_exit(void)
  114. {
  115. kmm_module_cleanup();
  116. }
  117. module_init(kmm_module_init);
  118. module_exit(kmm_module_exit);