sysfs-common.c 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Common Primitives for DAMON Sysfs Interface
  4. *
  5. * Author: SeongJae Park <[email protected]>
  6. */
  7. #include <linux/slab.h>
  8. #include "sysfs-common.h"
  9. DEFINE_MUTEX(damon_sysfs_lock);
  10. /*
  11. * unsigned long range directory
  12. */
  13. struct damon_sysfs_ul_range *damon_sysfs_ul_range_alloc(
  14. unsigned long min,
  15. unsigned long max)
  16. {
  17. struct damon_sysfs_ul_range *range = kmalloc(sizeof(*range),
  18. GFP_KERNEL);
  19. if (!range)
  20. return NULL;
  21. range->kobj = (struct kobject){};
  22. range->min = min;
  23. range->max = max;
  24. return range;
  25. }
  26. static ssize_t min_show(struct kobject *kobj, struct kobj_attribute *attr,
  27. char *buf)
  28. {
  29. struct damon_sysfs_ul_range *range = container_of(kobj,
  30. struct damon_sysfs_ul_range, kobj);
  31. return sysfs_emit(buf, "%lu\n", range->min);
  32. }
  33. static ssize_t min_store(struct kobject *kobj, struct kobj_attribute *attr,
  34. const char *buf, size_t count)
  35. {
  36. struct damon_sysfs_ul_range *range = container_of(kobj,
  37. struct damon_sysfs_ul_range, kobj);
  38. unsigned long min;
  39. int err;
  40. err = kstrtoul(buf, 0, &min);
  41. if (err)
  42. return err;
  43. range->min = min;
  44. return count;
  45. }
  46. static ssize_t max_show(struct kobject *kobj, struct kobj_attribute *attr,
  47. char *buf)
  48. {
  49. struct damon_sysfs_ul_range *range = container_of(kobj,
  50. struct damon_sysfs_ul_range, kobj);
  51. return sysfs_emit(buf, "%lu\n", range->max);
  52. }
  53. static ssize_t max_store(struct kobject *kobj, struct kobj_attribute *attr,
  54. const char *buf, size_t count)
  55. {
  56. struct damon_sysfs_ul_range *range = container_of(kobj,
  57. struct damon_sysfs_ul_range, kobj);
  58. unsigned long max;
  59. int err;
  60. err = kstrtoul(buf, 0, &max);
  61. if (err)
  62. return err;
  63. range->max = max;
  64. return count;
  65. }
  66. void damon_sysfs_ul_range_release(struct kobject *kobj)
  67. {
  68. kfree(container_of(kobj, struct damon_sysfs_ul_range, kobj));
  69. }
  70. static struct kobj_attribute damon_sysfs_ul_range_min_attr =
  71. __ATTR_RW_MODE(min, 0600);
  72. static struct kobj_attribute damon_sysfs_ul_range_max_attr =
  73. __ATTR_RW_MODE(max, 0600);
  74. static struct attribute *damon_sysfs_ul_range_attrs[] = {
  75. &damon_sysfs_ul_range_min_attr.attr,
  76. &damon_sysfs_ul_range_max_attr.attr,
  77. NULL,
  78. };
  79. ATTRIBUTE_GROUPS(damon_sysfs_ul_range);
  80. struct kobj_type damon_sysfs_ul_range_ktype = {
  81. .release = damon_sysfs_ul_range_release,
  82. .sysfs_ops = &kobj_sysfs_ops,
  83. .default_groups = damon_sysfs_ul_range_groups,
  84. };