Files
android_kernel_samsung_sm8750/drivers/cpufreq/qcom-cpufreq-thermal.c
2025-08-12 22:16:57 +02:00

213 lines
5.9 KiB
C
Executable File

// SPDX-License-Identifier: GPL-2.0-only
/*
* Copyright (c) 2024, Qualcomm Innovation Center, Inc. All rights reserved.
*/
#include <linux/cpufreq.h>
#include <linux/io.h>
#include <linux/mailbox_client.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/platform_device.h>
#if IS_ENABLED(CONFIG_SEC_PM_LOG)
#include <linux/sec_pm_log.h>
#endif
#include <trace/events/power.h>
struct qcom_cpufreq_thermal_domain {
struct mbox_client cl;
struct mbox_chan *ch;
struct cpufreq_policy *policy;
struct device_attribute freq_limit_attr;
unsigned long freq_limit;
#if IS_ENABLED(CONFIG_SEC_PM_LOG)
unsigned long lowest_freq;
bool limiting;
ktime_t start_time;
ktime_t limited_time;
unsigned long accu_time;
#endif
};
struct qcom_cpufreq_thermal {
struct qcom_cpufreq_thermal_domain *domains;
int num_domains;
};
static struct qcom_cpufreq_thermal qcom_cpufreq_thermal;
static inline
struct qcom_cpufreq_thermal_domain *to_qcom_cpufreq_thermal_domain(struct mbox_client *cl)
{
return container_of(cl, struct qcom_cpufreq_thermal_domain, cl);
}
static void qcom_cpufreq_thermal_rx(struct mbox_client *cl, void *msg)
{
struct qcom_cpufreq_thermal_domain *domain = to_qcom_cpufreq_thermal_domain(cl);
unsigned int cpu = cpumask_first(domain->policy->related_cpus);
unsigned long throttled_freq = *((unsigned long *)msg);
char lmh_debug[8] = {0};
dev_dbg(cl->dev, "cpu%u thermal limit: %lu\n", cpu, throttled_freq);
domain->freq_limit = throttled_freq;
#if IS_ENABLED(CONFIG_SEC_PM_LOG)
if (domain->limiting == false) {
ss_dcvsh_print("Start lmh cpu%d @%lu\n", cpu, (throttled_freq / 1000));
domain->lowest_freq = throttled_freq;
domain->limiting = true;
domain->start_time = ktime_get();
} else if (domain->limiting == true) {
if (throttled_freq >= domain->policy->cpuinfo.max_freq) {
domain->limiting = false;
domain->limited_time = (ktime_get() - domain->start_time);
domain->accu_time += ktime_to_ms(domain->limited_time);
ss_dcvsh_print("Fin. lmh cpu%d, lowest %lu, f_lim %lu, dcvsh %lu, accu %d\n",
cpu, (domain->lowest_freq / 1000), (throttled_freq / 1000),
(domain->policy->cur / 1000), domain->accu_time);
domain->lowest_freq = UINT_MAX;
} else {
if (throttled_freq < domain->lowest_freq)
domain->lowest_freq = throttled_freq;
}
}
#endif
arch_update_thermal_pressure(domain->policy->related_cpus, throttled_freq);
snprintf(lmh_debug, sizeof(lmh_debug), "lmh_%d", cpu);
trace_clock_set_rate(lmh_debug, throttled_freq, raw_smp_processor_id());
}
static ssize_t dcvsh_freq_limit_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct qcom_cpufreq_thermal_domain *domain;
domain = container_of(attr, struct qcom_cpufreq_thermal_domain, freq_limit_attr);
return scnprintf(buf, PAGE_SIZE, "%lu\n", domain->freq_limit);
}
static int qcom_cpufreq_thermal_driver_probe(struct platform_device *pdev)
{
struct qcom_cpufreq_thermal *data = &qcom_cpufreq_thermal;
struct qcom_cpufreq_thermal_domain *domain;
struct device *dev = &pdev->dev;
struct device *cpu_dev;
struct device_node *np = dev->of_node;
int ret, cpu, i;
data->num_domains = of_property_count_u32_elems(np, "qcom,policy-cpus");
if (data->num_domains < 0) {
dev_err(dev, "Error getting number of policies: %d\n", data->num_domains);
return data->num_domains;
}
data->domains = devm_kzalloc(dev, sizeof(*domain) * data->num_domains, GFP_KERNEL);
if (!data->domains)
return -ENOMEM;
for (i = 0; i < data->num_domains; i++) {
domain = &data->domains[i];
ret = of_property_read_u32_index(np, "qcom,policy-cpus", i, &cpu);
if (ret) {
dev_err(dev, "Error getting policy%d CPU: %d\n", i, ret);
goto err;
}
domain->policy = cpufreq_cpu_get(cpu);
if (!domain->policy) {
dev_dbg(dev, "Error getting policy for CPU%d\n", i);
ret = -EPROBE_DEFER;
goto err;
}
domain->cl.dev = dev;
domain->cl.rx_callback = qcom_cpufreq_thermal_rx;
domain->ch = mbox_request_channel(&domain->cl, i);
if (IS_ERR(domain->ch)) {
ret = PTR_ERR(domain->ch);
if (ret != -EPROBE_DEFER)
dev_err(dev, "Error getting mailbox %d: %d\n", i, ret);
goto err;
}
cpu_dev = get_cpu_device(cpu);
if (!cpu_dev) {
dev_err(dev, "Error getting CPU%d device\n", i);
ret = -EINVAL;
goto err;
}
sysfs_attr_init(&domain->freq_limit_attr.attr);
domain->freq_limit_attr.attr.name = "dcvsh_freq_limit";
domain->freq_limit_attr.show = dcvsh_freq_limit_show;
domain->freq_limit_attr.attr.mode = 0444;
domain->freq_limit = U32_MAX;
device_create_file(cpu_dev, &domain->freq_limit_attr);
}
dev_info(dev, "Probe successful\n");
return 0;
err:
for (i = 0; i < data->num_domains; i++) {
domain = &data->domains[i];
if (domain->policy)
cpufreq_cpu_put(domain->policy);
if (!IS_ERR_OR_NULL(domain->ch))
mbox_free_channel(domain->ch);
}
return ret;
}
static void qcom_cpufreq_thermal_driver_remove(struct platform_device *pdev)
{
struct qcom_cpufreq_thermal *data = &qcom_cpufreq_thermal;
struct qcom_cpufreq_thermal_domain *domain;
struct device *cpu_dev;
int i;
for (i = 0; i < data->num_domains; i++) {
domain = &data->domains[i];
mbox_free_channel(domain->ch);
cpu_dev = get_cpu_device(domain->policy->cpu);
device_remove_file(cpu_dev, &domain->freq_limit_attr);
cpufreq_cpu_put(domain->policy);
}
}
static const struct of_device_id qcom_cpufreq_thermal_match[] = {
{ .compatible = "qcom,cpufreq-thermal" },
{}
};
MODULE_DEVICE_TABLE(of, qcom_cpufreq_thermal_match);
static struct platform_driver qcom_cpufreq_thermal_driver = {
.probe = qcom_cpufreq_thermal_driver_probe,
.remove_new = qcom_cpufreq_thermal_driver_remove,
.driver = {
.name = "qcom-cpufreq-thermal",
.of_match_table = qcom_cpufreq_thermal_match,
},
};
static int __init qcom_cpufreq_thermal_init(void)
{
return platform_driver_register(&qcom_cpufreq_thermal_driver);
}
postcore_initcall(qcom_cpufreq_thermal_init);
MODULE_DESCRIPTION("QCOM CPUFREQ Thermal Driver");
MODULE_LICENSE("GPL");