123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * Copyright (c) 2010-2015, 2017, 2019-2021, The Linux Foundation. All rights reserved.
- * Copyright (c) 2023 Qualcomm Innovation Center, Inc. All rights reserved.
- */
- #include <linux/init.h>
- #include <linux/cpu.h>
- #include <linux/kobject.h>
- #include <linux/sysfs.h>
- #include <linux/module.h>
- #include <trace/hooks/sched.h>
- #define MAX_LONG_SIZE 24
- #define DEFAULT_DEF_TIMER_JIFFIES 5
- struct rq_data {
- unsigned long def_timer_jiffies;
- unsigned long def_timer_last_jiffy;
- int64_t def_start_time;
- struct attribute_group *attr_group;
- struct kobject *kobj;
- struct work_struct def_timer_work;
- };
- static struct rq_data rq_info;
- static struct workqueue_struct *rq_wq;
- spinlock_t rq_lock;
- static void wakeup_user(void *ignore, void *extra)
- {
- unsigned long jiffy_gap;
- jiffy_gap = jiffies - rq_info.def_timer_last_jiffy;
- if (jiffy_gap >= rq_info.def_timer_jiffies) {
- rq_info.def_timer_last_jiffy = jiffies;
- queue_work(rq_wq, &rq_info.def_timer_work);
- }
- }
- static void def_work_fn(struct work_struct *work)
- {
- /* Notify polling threads on change of value */
- sysfs_notify(rq_info.kobj, NULL, "def_timer_ms");
- }
- static ssize_t show_def_timer_ms(struct kobject *kobj,
- struct kobj_attribute *attr, char *buf)
- {
- int64_t diff;
- unsigned int udiff;
- diff = ktime_to_ns(ktime_get()) - rq_info.def_start_time;
- do_div(diff, 1000 * 1000);
- udiff = (unsigned int) diff;
- return scnprintf(buf, MAX_LONG_SIZE, "%u\n", udiff);
- }
- static ssize_t store_def_timer_ms(struct kobject *kobj,
- struct kobj_attribute *attr, const char *buf, size_t count)
- {
- unsigned int val = 0;
- if (kstrtouint(buf, 0, &val))
- return -EINVAL;
- rq_info.def_timer_jiffies = msecs_to_jiffies(val);
- rq_info.def_start_time = ktime_to_ns(ktime_get());
- return count;
- }
- static struct kobj_attribute def_timer_ms_attr =
- __ATTR(def_timer_ms, 0600, show_def_timer_ms,
- store_def_timer_ms);
- static struct attribute *rq_attrs[] = {
- &def_timer_ms_attr.attr,
- NULL,
- };
- static struct attribute_group rq_attr_group = {
- .attrs = rq_attrs,
- };
- static int init_rq_attribs(void)
- {
- int err;
- rq_info.attr_group = &rq_attr_group;
- /* Create /sys/devices/system/cpu/cpu0/rq-stats/... */
- rq_info.kobj = kobject_create_and_add("rq-stats",
- &get_cpu_device(0)->kobj);
- if (!rq_info.kobj)
- return -ENOMEM;
- err = sysfs_create_group(rq_info.kobj, rq_info.attr_group);
- if (err)
- kobject_put(rq_info.kobj);
- else
- kobject_uevent(rq_info.kobj, KOBJ_ADD);
- return err;
- }
- static int __init msm_rq_stats_init(void)
- {
- int ret;
- #ifndef CONFIG_SMP
- /* Bail out if this is not an SMP Target */
- return -EPERM;
- #endif
- rq_wq = create_singlethread_workqueue("rq_stats");
- WARN_ON(!rq_wq);
- INIT_WORK(&rq_info.def_timer_work, def_work_fn);
- spin_lock_init(&rq_lock);
- rq_info.def_timer_jiffies = DEFAULT_DEF_TIMER_JIFFIES;
- rq_info.def_timer_last_jiffy = 0;
- ret = init_rq_attribs();
- register_trace_android_vh_jiffies_update(wakeup_user, NULL);
- return ret;
- }
- late_initcall(msm_rq_stats_init);
- MODULE_DESCRIPTION("QCOM Run Queue Stats");
- MODULE_LICENSE("GPL v2");
|