// SPDX-License-Identifier: GPL-2.0 /* * mediatek-mcupm-cpufreq.c - MCUPM based CPUFreq Driver * * Copyright (c) 2020 MediaTek Inc. * Wei-Chia Su */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "../misc/mediatek/mcupm/include/mcupm_driver.h" #include "../misc/mediatek/mcupm/include/mcupm_ipi_id.h" #define OFFS_WFI_S 0x037c #define DVFS_D_LEN (4) #define UDATA_OFF 0x10 #define UDATA_SIZE 0x120 enum { IPI_EEMSN_SHARERAM_INIT, }; enum cpu_dvfs_ipi_type { IPI_DVFS_INIT, NR_DVFS_IPI, }; struct ipi_data { unsigned int cmd; union { struct { unsigned int arg[3]; } set_fv; } u; }; struct mtk_cpu_dvfs_info { struct cpumask cpus; struct device *cpu_dev; struct list_head list_head; struct mutex lock; void __iomem *csram_base; void __iomem *usram_base; unsigned int cluster; }; struct mtk_cpufreq_match_data { int (*get_version)(struct platform_device *pdev, unsigned int *arr); }; static LIST_HEAD(dvfs_info_list); static struct mtk_cpu_dvfs_info *mtk_cpu_dvfs_info_lookup(int cpu) { struct mtk_cpu_dvfs_info *info; struct list_head *list; list_for_each(list, &dvfs_info_list) { info = list_entry(list, struct mtk_cpu_dvfs_info, list_head); if (cpumask_test_cpu(cpu, &info->cpus)) return info; } return NULL; } static struct mtk_cpu_dvfs_info *mtk_cpu_dvfs_info_lookup_by_dev(struct device *cpu_dev) { struct mtk_cpu_dvfs_info *info; struct list_head *list; list_for_each(list, &dvfs_info_list) { info = list_entry(list, struct mtk_cpu_dvfs_info, list_head); if (info->cpu_dev == cpu_dev) return info; } return NULL; } static int mtk_cpufreq_set_target(struct cpufreq_policy *policy, unsigned int index) { struct mtk_cpu_dvfs_info *info = policy->driver_data; unsigned int cluster_id = info->cluster; writel_relaxed(index, info->csram_base + (OFFS_WFI_S + (cluster_id * 4)) ); arch_set_freq_scale(policy->related_cpus, policy->freq_table[index].frequency, policy->cpuinfo.max_freq); return 0; } static int mtk_cpu_dvfs_info_init(struct mtk_cpu_dvfs_info *info, int cpu) { int ret; /* Get OPP-sharing information from "operating-points-v2" bindings */ ret = dev_pm_opp_of_get_sharing_cpus(info->cpu_dev, &info->cpus); if (ret) goto out_free_resources; ret = dev_pm_opp_of_cpumask_add_table(&info->cpus); if (ret) goto out_free_resources; mutex_init(&info->lock); return 0; out_free_resources: return ret; } static void mtk_cpu_dvfs_info_release(struct mtk_cpu_dvfs_info *info) { dev_pm_opp_of_cpumask_remove_table(&info->cpus); } int mcupm_get_cpu_power(unsigned long *power, unsigned long *KHz, struct device *cpu_dev) { struct mtk_cpu_dvfs_info *info = mtk_cpu_dvfs_info_lookup_by_dev(cpu_dev); unsigned long Hz; int ret; struct dev_pm_opp *opp; int level; if (!cpu_dev) { pr_info("failed to get device\n"); return -ENODEV; } /* Get the power cost of the performance domain. */ Hz = *KHz * 1000; opp = dev_pm_opp_find_freq_ceil(cpu_dev, &Hz); level = dev_pm_opp_get_level(opp); /* The EM framework specifies the frequency in KHz. */ *power = readl(info->usram_base + level * DVFS_D_LEN) / 1000; *KHz = Hz / 1000; return 0; } static int mtk_cpufreq_init(struct cpufreq_policy *policy) { struct mtk_cpu_dvfs_info *info; struct cpufreq_frequency_table *freq_table; int ret, opp; struct em_data_callback em_cb = EM_DATA_CB(mcupm_get_cpu_power); info = mtk_cpu_dvfs_info_lookup(policy->cpu); if (!info) return -EINVAL; ret = dev_pm_opp_init_cpufreq_table(info->cpu_dev, &freq_table); if (ret) return ret; cpumask_copy(policy->cpus, &info->cpus); policy->driver_data = info; policy->freq_table = freq_table; policy->transition_delay_us = 1000; /* us */ opp = dev_pm_opp_get_opp_count(info->cpu_dev); em_dev_register_perf_domain(info->cpu_dev, opp, &em_cb, policy->cpus); return 0; out_free_opp: dev_pm_opp_of_cpumask_remove_table(policy->cpus); out_free_cpufreq_table: dev_pm_opp_free_cpufreq_table(info->cpu_dev, &freq_table); return ret; } static int mtk_cpufreq_exit(struct cpufreq_policy *policy) { struct mtk_cpu_dvfs_info *info = policy->driver_data; dev_pm_opp_free_cpufreq_table(info->cpu_dev, &policy->freq_table); return 0; } static struct cpufreq_driver mtk_cpufreq_driver = { .flags = CPUFREQ_STICKY | CPUFREQ_NEED_INITIAL_FREQ_CHECK | CPUFREQ_HAVE_GOVERNOR_PER_POLICY | CPUFREQ_IS_COOLING_DEV, .verify = cpufreq_generic_frequency_table_verify, .target_index = mtk_cpufreq_set_target, .init = mtk_cpufreq_init, .exit = mtk_cpufreq_exit, .name = "mtk-cpufreq", .attr = cpufreq_generic_attr, }; static int mt6873_get_version(struct platform_device *pdev, unsigned int *arr) { const char *nvmem_name_arr[2] = {"seg_code", "level_code1"}; int ret, i, version; unsigned int seg_reg[2]; for (i = 0; i < 2; i++) { ret = nvmem_cell_read_u32(&pdev->dev, nvmem_name_arr[i], &seg_reg[i]); if (ret) break; } arr[0] = seg_reg[0] << 4 | 1; arr[1] = (seg_reg[1] & 0xFF) > 2 ? 0x10 : 0x1; arr[2] = (seg_reg[1] & 0x80000000) ? 0x10 : 0x1; return 3; } static int mtk_cpufreq_probe(struct platform_device *pdev) { struct mtk_cpu_dvfs_info *info; struct list_head *list, *tmp; int cpu, ret, sig, count; int cluster = 0; struct ipi_data cdvfs_d, eem_data; uint32_t cpufreq_buf[4]; struct mtk_ipi_device *mcupm_ipidev, *mcupm_eem_ipidev; int ipi_ackdata; struct device *cpu_dev; unsigned int version[3]; const struct of_device_id *match; struct mtk_cpufreq_match_data *data; mcupm_ipidev = (struct mtk_ipi_device *) get_mcupm_ipidev(); mcupm_eem_ipidev = (struct mtk_ipi_device *) get_mcupm_ipidev(); if (!mcupm_ipidev) return -ENODEV; ret = mtk_ipi_register(mcupm_ipidev, CH_S_CPU_DVFS, NULL, NULL, (void *) &cpufreq_buf); if (ret) return -EINVAL; cdvfs_d.cmd = IPI_DVFS_INIT; cdvfs_d.u.set_fv.arg[0] = 1; ret = mtk_ipi_send_compl(mcupm_ipidev, CH_S_CPU_DVFS, IPI_SEND_POLLING, &cdvfs_d, sizeof(struct ipi_data)/MBOX_SLOT_SIZE, 2000); if (ret) return -EINVAL; ret = mtk_ipi_register(mcupm_eem_ipidev, CH_S_EEMSN, NULL, NULL, (void *)&ipi_ackdata); if (ret != 0) return -EINVAL; eem_data.cmd = IPI_EEMSN_SHARERAM_INIT; ret = mtk_ipi_send_compl(mcupm_eem_ipidev, CH_S_EEMSN, IPI_SEND_POLLING, &eem_data, sizeof(struct ipi_data)/MBOX_SLOT_SIZE, 2000); if (ret) return -EINVAL; data = (struct mtk_cpufreq_match_data *)of_device_get_match_data(&pdev->dev); count = 0; if (data) { if (data->get_version) count = data->get_version(pdev, version); } for_each_possible_cpu(cpu) { info = mtk_cpu_dvfs_info_lookup(cpu); if (info) continue; info = devm_kzalloc(&pdev->dev, sizeof(*info), GFP_KERNEL); if (!info) { ret = -ENOMEM; goto release_dvfs_info_list; } info->csram_base = of_iomap(pdev->dev.of_node, 0); if (!info->csram_base) { ret = -ENOMEM; goto release_dvfs_info_list; } cpu_dev = get_cpu_device(cpu); if (!cpu_dev) return -ENODEV; info->cpu_dev = cpu_dev; if (count) dev_pm_opp_set_supported_hw(cpu_dev, version, count); ret = mtk_cpu_dvfs_info_init(info, cpu); if (ret) goto release_dvfs_info_list; info->cluster = cluster; info->usram_base = of_iomap(pdev->dev.of_node, 1); info->usram_base += (UDATA_OFF + (cluster) * UDATA_SIZE); cluster++; mutex_init(&info->lock); list_add(&info->list_head, &dvfs_info_list); } /* using last info to check */ if (readl_poll_timeout( (info->usram_base - (info->cluster) * UDATA_SIZE - UDATA_OFF), sig, (sig == 0x5A5A5A02), 1000, 30000)) pr_info("volt no updated\n"); ret = cpufreq_register_driver(&mtk_cpufreq_driver); if (ret) goto release_dvfs_info_list; return 0; release_dvfs_info_list: list_for_each_safe(list, tmp, &dvfs_info_list) { info = list_entry(list, struct mtk_cpu_dvfs_info, list_head); mtk_cpu_dvfs_info_release(info); list_del(list); } return ret; } static const struct mtk_cpufreq_match_data mt6873_cpufreq_data = { .get_version = mt6873_get_version, }; /* List of machines supported by this driver */ static const struct of_device_id mtk_cpufreq_machines[] = { { .compatible = "mediatek,mt6873-mcupm-dvfs", .data = &mt6873_cpufreq_data }, { } }; MODULE_DEVICE_TABLE(of, mtk_cpufreq_machines); static struct platform_driver mtk_cpufreq_platdrv = { .probe = mtk_cpufreq_probe, .driver = { .name = "dvfsp", .of_match_table = of_match_ptr(mtk_cpufreq_machines), }, }; module_platform_driver(mtk_cpufreq_platdrv); MODULE_AUTHOR("Wei-Chia Su "); MODULE_DESCRIPTION("Medaitek MCUPM CPUFreq Platform driver"); MODULE_LICENSE("GPL v2");