// SPDX-License-Identifier: GPL-2.0 /* * Copyright (c) 2019 MediaTek Inc. */ #include #include #include #include #include #include #include #include #include #include #include #include #include "dvfsrc-met.h" #include "dvfsrc-met.h" static struct mtk_dvfsrc_met *dvfsrc_drv; enum met_info_index { INFO_OPP_IDX = 0, INFO_FREQ_IDX, INFO_VCORE_IDX, INFO_SPM_LEVEL_IDX, INFO_MAX, }; static unsigned int met_vcorefs_info[INFO_MAX]; static char *met_info_name[INFO_MAX] = { "OPP", "FREQ", "VCORE", "x__SPM_LEVEL", }; int vcorefs_get_num_opp(void) { return mtk_dvfsrc_query_opp_info(MTK_DVFSRC_NUM_DVFS_OPP); } EXPORT_SYMBOL(vcorefs_get_num_opp); int vcorefs_get_opp_info_num(void) { return INFO_MAX; } EXPORT_SYMBOL(vcorefs_get_opp_info_num); char **vcorefs_get_opp_info_name(void) { return met_info_name; } EXPORT_SYMBOL(vcorefs_get_opp_info_name); unsigned int *vcorefs_get_opp_info(void) { struct mtk_dvfsrc_met *dvfsrc = dvfsrc_drv; if (dvfsrc) { met_vcorefs_info[INFO_OPP_IDX] = mtk_dvfsrc_query_opp_info(MTK_DVFSRC_CURR_DVFS_OPP); met_vcorefs_info[INFO_FREQ_IDX] = mtk_dvfsrc_query_opp_info(MTK_DVFSRC_CURR_DRAM_KHZ); met_vcorefs_info[INFO_VCORE_IDX] = mtk_dvfsrc_query_opp_info(MTK_DVFSRC_CURR_VCORE_UV); met_vcorefs_info[INFO_SPM_LEVEL_IDX] = dvfsrc->dvd->met->get_current_level(dvfsrc); } return met_vcorefs_info; } EXPORT_SYMBOL(vcorefs_get_opp_info); int vcorefs_get_src_req_num(void) { struct mtk_dvfsrc_met *dvfsrc = dvfsrc_drv; if (dvfsrc) return dvfsrc->dvd->met->dvfsrc_get_src_req_num(); return 0; } EXPORT_SYMBOL(vcorefs_get_src_req_num); char **vcorefs_get_src_req_name(void) { struct mtk_dvfsrc_met *dvfsrc = dvfsrc_drv; if (dvfsrc) return dvfsrc->dvd->met->dvfsrc_get_src_req_name(); return NULL; } EXPORT_SYMBOL(vcorefs_get_src_req_name); unsigned int *vcorefs_get_src_req(void) { struct mtk_dvfsrc_met *dvfsrc = dvfsrc_drv; if (dvfsrc) return dvfsrc->dvd->met->dvfsrc_get_src_req(dvfsrc); return NULL; } EXPORT_SYMBOL(vcorefs_get_src_req); int get_cur_ddr_ratio(void) { struct mtk_dvfsrc_met *dvfsrc = dvfsrc_drv; if (dvfsrc) return dvfsrc->dvd->met->dvfsrc_get_ddr_ratio(dvfsrc); return 0; } EXPORT_SYMBOL(get_cur_ddr_ratio); static ssize_t dvfsrc_met_dump_show(struct device *dev, struct device_attribute *attr, char *buf) { char *p = buf; char *buff_end = p + PAGE_SIZE; char **name; unsigned int *value; int i, res_num; p += snprintf(p, buff_end - p, "NUM_VCORE_OPP : %d\n", vcorefs_get_num_opp()); res_num = vcorefs_get_opp_info_num(); name = vcorefs_get_opp_info_name(); value = vcorefs_get_opp_info(); p += snprintf(p, buff_end - p, "NUM_OPP_INFO : %d\n", res_num); for (i = 0; i < res_num; i++) { p += snprintf(p, buff_end - p, "%s : %d\n", name[i], value[i]); } res_num = vcorefs_get_src_req_num(); name = vcorefs_get_src_req_name(); value = vcorefs_get_src_req(); p += snprintf(p, buff_end - p, "NUM SRC_REQ: %d\n", res_num); if (name && value) { for (i = 0; i < res_num; i++) { p += snprintf(p, buff_end - p, "%s : %d\n", name[i], value[i]); } } return p - buf; } static DEVICE_ATTR_RO(dvfsrc_met_dump); static struct attribute *dvfsrc_attrs[] = { &dev_attr_dvfsrc_met_dump.attr, NULL, }; static struct attribute_group dvfsrc_met_attr_group = { .attrs = dvfsrc_attrs, }; static int dvfsrc_met_register_sysfs(struct device *dev) { int ret; ret = sysfs_create_group(&dev->kobj, &dvfsrc_met_attr_group); if (ret) return ret; ret = sysfs_create_link(&dev->parent->kobj, &dev->kobj, "met"); return ret; } static void dvfsrc_met_unregister_sysfs(struct device *dev) { sysfs_remove_link(&dev->parent->kobj, "met"); sysfs_remove_group(&dev->kobj, &dvfsrc_met_attr_group); } static int mtk_dvfsrc_met_probe(struct platform_device *pdev) { struct device *dev = &pdev->dev; struct platform_device *parent_dev; struct resource *res; struct mtk_dvfsrc_met *dvfsrc; parent_dev = to_platform_device(dev->parent); dvfsrc = devm_kzalloc(&pdev->dev, sizeof(*dvfsrc), GFP_KERNEL); if (!dvfsrc) return -ENOMEM; dvfsrc->dvd = of_device_get_match_data(&pdev->dev); dvfsrc->dev = &pdev->dev; res = platform_get_resource_byname(parent_dev, IORESOURCE_MEM, "dvfsrc"); if (!res) { dev_info(dev, "dvfsrc debug resource not found\n"); return -ENODEV; } dvfsrc->regs = devm_ioremap(&pdev->dev, res->start, resource_size(res)); if (IS_ERR(dvfsrc->regs)) return PTR_ERR(dvfsrc->regs); dvfsrc_drv = dvfsrc; platform_set_drvdata(pdev, dvfsrc); dvfsrc_met_register_sysfs(dev); return 0; } static const struct dvfsrc_met_data mt6779_data = { .met = &mt6779_met_config, }; static const struct dvfsrc_met_data mt6761_data = { .met = &mt6761_met_config, }; static int mtk_dvfsrc_met_remove(struct platform_device *pdev) { struct device *dev = &pdev->dev; dvfsrc_met_unregister_sysfs(dev); dvfsrc_drv = NULL; return 0; } static const struct of_device_id mtk_dvfsrc_met_of_match[] = { { .compatible = "mediatek,mt6779-dvfsrc-met", .data = &mt6779_data, }, { .compatible = "mediatek,mt6761-dvfsrc-met", .data = &mt6761_data, }, { .compatible = "mediatek,mt6765-dvfsrc-met", .data = &mt6761_data, }, { /* sentinel */ }, }; static struct platform_driver mtk_dvfsrc_met_driver = { .probe = mtk_dvfsrc_met_probe, .remove = mtk_dvfsrc_met_remove, .driver = { .name = "mtk-dvfsrc-met", .of_match_table = of_match_ptr(mtk_dvfsrc_met_of_match), }, }; static int __init mtk_dvfsrc_met_init(void) { return platform_driver_register(&mtk_dvfsrc_met_driver); } late_initcall_sync(mtk_dvfsrc_met_init) static void __exit mtk_dvfsrc_met_exit(void) { platform_driver_unregister(&mtk_dvfsrc_met_driver); } module_exit(mtk_dvfsrc_met_exit); MODULE_LICENSE("GPL v2"); MODULE_DESCRIPTION("MTK DVFSRC met driver");