mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-22 16:06:04 -05:00
9265ea04a7
No functional modification involved. ./drivers/interconnect/mediatek/icc-emi.c:71:2-3: Unneeded semicolon. Reported-by: Abaci Robot <abaci@linux.alibaba.com> Closes: https://bugzilla.openanolis.cn/show_bug.cgi?id=9401 Signed-off-by: Jiapeng Chong <jiapeng.chong@linux.alibaba.com> Link: https://lore.kernel.org/r/20240624025326.69551-1-jiapeng.chong@linux.alibaba.com Signed-off-by: Georgi Djakov <djakov@kernel.org>
153 lines
3.5 KiB
C
153 lines
3.5 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* MediaTek External Memory Interface (EMI) Interconnect driver
|
|
*
|
|
* Copyright (c) 2021 MediaTek Inc.
|
|
* Copyright (c) 2024 Collabora Ltd.
|
|
* AngeloGioacchino Del Regno <angelogioacchino.delregno@collabora.com>
|
|
*/
|
|
|
|
#include <linux/interconnect.h>
|
|
#include <linux/interconnect-provider.h>
|
|
#include <linux/module.h>
|
|
#include <linux/of.h>
|
|
#include <linux/of_platform.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/soc/mediatek/dvfsrc.h>
|
|
|
|
#include "icc-emi.h"
|
|
|
|
static int mtk_emi_icc_aggregate(struct icc_node *node, u32 tag, u32 avg_bw,
|
|
u32 peak_bw, u32 *agg_avg, u32 *agg_peak)
|
|
{
|
|
struct mtk_icc_node *in = node->data;
|
|
|
|
*agg_avg += avg_bw;
|
|
*agg_peak = max_t(u32, *agg_peak, peak_bw);
|
|
|
|
in->sum_avg = *agg_avg;
|
|
in->max_peak = *agg_peak;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int mtk_emi_icc_set(struct icc_node *src, struct icc_node *dst)
|
|
{
|
|
struct mtk_icc_node *node = dst->data;
|
|
struct device *dev;
|
|
int ret;
|
|
|
|
if (unlikely(!src->provider))
|
|
return -EINVAL;
|
|
|
|
dev = src->provider->dev;
|
|
|
|
switch (node->ep) {
|
|
case 0:
|
|
break;
|
|
case 1:
|
|
ret = mtk_dvfsrc_send_request(dev, MTK_DVFSRC_CMD_PEAK_BW, node->max_peak);
|
|
if (ret) {
|
|
dev_err(dev, "Cannot send peak bw request: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
ret = mtk_dvfsrc_send_request(dev, MTK_DVFSRC_CMD_BW, node->sum_avg);
|
|
if (ret) {
|
|
dev_err(dev, "Cannot send bw request: %d\n", ret);
|
|
return ret;
|
|
}
|
|
break;
|
|
case 2:
|
|
ret = mtk_dvfsrc_send_request(dev, MTK_DVFSRC_CMD_HRT_BW, node->sum_avg);
|
|
if (ret) {
|
|
dev_err(dev, "Cannot send HRT bw request: %d\n", ret);
|
|
return ret;
|
|
}
|
|
break;
|
|
default:
|
|
dev_err(src->provider->dev, "Unknown endpoint %u\n", node->ep);
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int mtk_emi_icc_probe(struct platform_device *pdev)
|
|
{
|
|
const struct mtk_icc_desc *desc;
|
|
struct device *dev = &pdev->dev;
|
|
struct icc_node *node;
|
|
struct icc_onecell_data *data;
|
|
struct icc_provider *provider;
|
|
struct mtk_icc_node **mnodes;
|
|
int i, j, ret;
|
|
|
|
desc = of_device_get_match_data(dev);
|
|
if (!desc)
|
|
return -EINVAL;
|
|
|
|
mnodes = desc->nodes;
|
|
|
|
provider = devm_kzalloc(dev, sizeof(*provider), GFP_KERNEL);
|
|
if (!provider)
|
|
return -ENOMEM;
|
|
|
|
data = devm_kzalloc(dev, struct_size(data, nodes, desc->num_nodes), GFP_KERNEL);
|
|
if (!data)
|
|
return -ENOMEM;
|
|
|
|
provider->dev = pdev->dev.parent;
|
|
provider->set = mtk_emi_icc_set;
|
|
provider->aggregate = mtk_emi_icc_aggregate;
|
|
provider->xlate = of_icc_xlate_onecell;
|
|
INIT_LIST_HEAD(&provider->nodes);
|
|
provider->data = data;
|
|
|
|
for (i = 0; i < desc->num_nodes; i++) {
|
|
if (!mnodes[i])
|
|
continue;
|
|
|
|
node = icc_node_create(mnodes[i]->id);
|
|
if (IS_ERR(node)) {
|
|
ret = PTR_ERR(node);
|
|
goto err;
|
|
}
|
|
|
|
node->name = mnodes[i]->name;
|
|
node->data = mnodes[i];
|
|
icc_node_add(node, provider);
|
|
|
|
for (j = 0; j < mnodes[i]->num_links; j++)
|
|
icc_link_create(node, mnodes[i]->links[j]);
|
|
|
|
data->nodes[i] = node;
|
|
}
|
|
data->num_nodes = desc->num_nodes;
|
|
|
|
ret = icc_provider_register(provider);
|
|
if (ret)
|
|
goto err;
|
|
|
|
platform_set_drvdata(pdev, provider);
|
|
|
|
return 0;
|
|
err:
|
|
icc_nodes_remove(provider);
|
|
return ret;
|
|
}
|
|
EXPORT_SYMBOL_GPL(mtk_emi_icc_probe);
|
|
|
|
void mtk_emi_icc_remove(struct platform_device *pdev)
|
|
{
|
|
struct icc_provider *provider = platform_get_drvdata(pdev);
|
|
|
|
icc_provider_deregister(provider);
|
|
icc_nodes_remove(provider);
|
|
}
|
|
EXPORT_SYMBOL_GPL(mtk_emi_icc_remove);
|
|
|
|
MODULE_AUTHOR("AngeloGioacchino Del Regno <angelogioacchino.delregno@collabora.com>");
|
|
MODULE_AUTHOR("Henry Chen <henryc.chen@mediatek.com>");
|
|
MODULE_DESCRIPTION("MediaTek External Memory Interface interconnect driver");
|
|
MODULE_LICENSE("GPL");
|