Files
2025-08-12 22:16:57 +02:00

222 lines
4.7 KiB
C
Executable File

// SPDX-License-Identifier: GPL-2.0-only
/*
* Copyright (c) 2021, The Linux Foundation. All rights reserved.
* Copyright (c) 2022-2023, Qualcomm Innovation Center, Inc. All rights reserved.
*/
#include <linux/cpu.h>
#include <linux/cpuidle.h>
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/pm_domain.h>
#include <linux/slab.h>
#include <linux/string.h>
#include "qcom-lpm.h"
static struct kobject *qcom_lpm_kobj;
static ssize_t cluster_idle_set(struct kobject *kobj,
struct kobj_attribute *attr,
const char *buf, size_t len)
{
struct qcom_cluster_node *d = container_of(attr, struct qcom_cluster_node, disable_attr);
bool disable;
int ret;
ret = strtobool(buf, &disable);
if (ret)
return -EINVAL;
d->cluster->state_allowed[d->state_idx] = !disable;
return len;
}
static ssize_t cluster_idle_get(struct kobject *kobj,
struct kobj_attribute *attr,
char *buf)
{
struct qcom_cluster_node *d = container_of(attr, struct qcom_cluster_node, disable_attr);
return scnprintf(buf, PAGE_SIZE, "%d\n", !d->cluster->state_allowed[d->state_idx]);
}
static int create_cluster_state_node(struct device *dev, struct qcom_cluster_node *d)
{
struct kobj_attribute *attr = &d->disable_attr;
int ret;
d->attr_group = devm_kzalloc(dev, sizeof(struct attribute_group), GFP_KERNEL);
if (!d->attr_group)
return -ENOMEM;
d->attrs = devm_kcalloc(dev, 2, sizeof(struct attribute *), GFP_KERNEL);
if (!d->attrs)
return -ENOMEM;
sysfs_attr_init(&attr->attr);
attr->attr.name = "disable";
attr->attr.mode = 0644;
attr->show = cluster_idle_get;
attr->store = cluster_idle_set;
d->attrs[0] = &attr->attr;
d->attrs[1] = NULL;
d->attr_group->attrs = d->attrs;
ret = sysfs_create_group(d->kobj, d->attr_group);
if (ret)
return -ENOMEM;
return ret;
}
void remove_cluster_sysfs_nodes(struct lpm_cluster *cluster)
{
struct generic_pm_domain *genpd = cluster->genpd;
struct kobject *kobj = cluster->dev_kobj;
int i;
if (!qcom_lpm_kobj)
return;
for (i = 0; i < genpd->state_count; i++) {
struct qcom_cluster_node *d = cluster->dev_node[i];
kobject_put(d->kobj);
}
kobject_put(kobj);
}
int create_cluster_sysfs_nodes(struct lpm_cluster *cluster)
{
char name[10];
int i, ret;
struct generic_pm_domain *genpd = cluster->genpd;
if (!qcom_lpm_kobj)
return -EPROBE_DEFER;
cluster->dev_kobj = kobject_create_and_add(genpd->name, qcom_lpm_kobj);
if (!cluster->dev_kobj)
return -ENOMEM;
for (i = 0; i < genpd->state_count; i++) {
struct qcom_cluster_node *d;
d = devm_kzalloc(cluster->dev, sizeof(*d), GFP_KERNEL);
if (!d) {
kobject_put(cluster->dev_kobj);
return -ENOMEM;
}
d->state_idx = i;
d->cluster = cluster;
scnprintf(name, PAGE_SIZE, "D%u", i);
d->kobj = kobject_create_and_add(name, cluster->dev_kobj);
if (!d->kobj) {
kobject_put(cluster->dev_kobj);
return -ENOMEM;
}
ret = create_cluster_state_node(cluster->dev, d);
if (ret) {
kobject_put(d->kobj);
kobject_put(cluster->dev_kobj);
return ret;
}
cluster->dev_node[i] = d;
}
return 0;
}
static ssize_t sleep_disabled_show(struct kobject *kobj,
struct kobj_attribute *attr,
char *buf)
{
return scnprintf(buf, PAGE_SIZE, "%u\n", sleep_disabled);
}
static ssize_t sleep_disabled_store(struct kobject *kobj,
struct kobj_attribute *attr,
const char *buf, size_t count)
{
bool val;
int ret;
ret = kstrtobool(buf, &val);
if (ret) {
pr_err("Invalid argument passed\n");
return ret;
}
sleep_disabled = val;
return count;
}
static ssize_t prediction_disabled_show(struct kobject *kobj,
struct kobj_attribute *attr,
char *buf)
{
return scnprintf(buf, PAGE_SIZE, "%u\n", prediction_disabled);
}
static ssize_t prediction_disabled_store(struct kobject *kobj,
struct kobj_attribute *attr,
const char *buf, size_t count)
{
bool val;
int ret;
ret = kstrtobool(buf, &val);
if (ret) {
pr_err("Invalid argument passed\n");
return ret;
}
prediction_disabled = val;
return count;
}
static struct kobj_attribute attr_sleep_disabled = __ATTR_RW(sleep_disabled);
static struct kobj_attribute attr_prediction_disabled = __ATTR_RW(prediction_disabled);
static struct attribute *lpm_gov_attrs[] = {
&attr_sleep_disabled.attr,
&attr_prediction_disabled.attr,
NULL
};
static struct attribute_group lpm_gov_attr_group = {
.attrs = lpm_gov_attrs,
.name = "parameters",
};
void remove_global_sysfs_nodes(void)
{
kobject_put(qcom_lpm_kobj);
}
int create_global_sysfs_nodes(void)
{
struct kobject *cpuidle_kobj;
struct device *dev_root = bus_get_dev_root(&cpu_subsys);
if (!dev_root)
return -EINVAL;
cpuidle_kobj = &dev_root->kobj;
qcom_lpm_kobj = kobject_create_and_add(KBUILD_MODNAME, cpuidle_kobj);
if (!qcom_lpm_kobj)
return -ENOMEM;
return sysfs_create_group(qcom_lpm_kobj, &lpm_gov_attr_group);
}