root/drivers/media/platform/qcom/iris/iris_resources.c
// SPDX-License-Identifier: GPL-2.0-only
/*
 * Copyright (c) 2022-2024 Qualcomm Innovation Center, Inc. All rights reserved.
 */

#include <linux/clk.h>
#include <linux/devfreq.h>
#include <linux/interconnect.h>
#include <linux/pm_domain.h>
#include <linux/pm_opp.h>
#include <linux/pm_runtime.h>
#include <linux/reset.h>

#include "iris_core.h"
#include "iris_resources.h"

#define BW_THRESHOLD 50000

int iris_set_icc_bw(struct iris_core *core, unsigned long icc_bw)
{
        unsigned long bw_kbps = 0, bw_prev = 0;
        const struct icc_info *icc_tbl;
        int ret = 0, i;

        icc_tbl = core->iris_platform_data->icc_tbl;

        for (i = 0; i < core->icc_count; i++) {
                if (!strcmp(core->icc_tbl[i].name, "video-mem")) {
                        bw_kbps = icc_bw;
                        bw_prev = core->power.icc_bw;

                        bw_kbps = clamp_t(typeof(bw_kbps), bw_kbps,
                                          icc_tbl[i].bw_min_kbps, icc_tbl[i].bw_max_kbps);

                        if (abs(bw_kbps - bw_prev) < BW_THRESHOLD && bw_prev)
                                return ret;

                        core->icc_tbl[i].avg_bw = bw_kbps;

                        core->power.icc_bw = bw_kbps;
                        break;
                }
        }

        return icc_bulk_set_bw(core->icc_count, core->icc_tbl);
}

int iris_unset_icc_bw(struct iris_core *core)
{
        u32 i;

        core->power.icc_bw = 0;

        for (i = 0; i < core->icc_count; i++) {
                core->icc_tbl[i].avg_bw = 0;
                core->icc_tbl[i].peak_bw = 0;
        }

        return icc_bulk_set_bw(core->icc_count, core->icc_tbl);
}

int iris_opp_set_rate(struct device *dev, unsigned long freq)
{
        struct dev_pm_opp *opp __free(put_opp);

        opp = devfreq_recommended_opp(dev, &freq, 0);
        if (IS_ERR(opp))
                return PTR_ERR(opp);

        return dev_pm_opp_set_opp(dev, opp);
}

int iris_enable_power_domains(struct iris_core *core, struct device *pd_dev)
{
        int ret;

        ret = iris_opp_set_rate(core->dev, ULONG_MAX);
        if (ret)
                return ret;

        ret = pm_runtime_get_sync(pd_dev);
        if (ret < 0)
                return ret;

        return ret;
}

int iris_disable_power_domains(struct iris_core *core, struct device *pd_dev)
{
        int ret;

        ret = iris_opp_set_rate(core->dev, 0);
        if (ret)
                return ret;

        pm_runtime_put_sync(pd_dev);

        return 0;
}

static struct clk *iris_get_clk_by_type(struct iris_core *core, enum platform_clk_type clk_type)
{
        const struct platform_clk_data *clk_tbl;
        u32 clk_cnt, i, j;

        clk_tbl = core->iris_platform_data->clk_tbl;
        clk_cnt = core->iris_platform_data->clk_tbl_size;

        for (i = 0; i < clk_cnt; i++) {
                if (clk_tbl[i].clk_type == clk_type) {
                        for (j = 0; core->clock_tbl && j < core->clk_count; j++) {
                                if (!strcmp(core->clock_tbl[j].id, clk_tbl[i].clk_name))
                                        return core->clock_tbl[j].clk;
                        }
                }
        }

        return NULL;
}

int iris_prepare_enable_clock(struct iris_core *core, enum platform_clk_type clk_type)
{
        struct clk *clock;

        clock = iris_get_clk_by_type(core, clk_type);
        if (!clock)
                return -ENOENT;

        return clk_prepare_enable(clock);
}

int iris_disable_unprepare_clock(struct iris_core *core, enum platform_clk_type clk_type)
{
        struct clk *clock;

        clock = iris_get_clk_by_type(core, clk_type);
        if (!clock)
                return -EINVAL;

        clk_disable_unprepare(clock);

        return 0;
}