202 lines
4.5 KiB
202 lines
4.5 KiB
/* Copyright (c) 2010-2015, 2017, The Linux Foundation. All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2 and
|
|
* only version 2 as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*/
|
|
|
|
#define pr_fmt(fmt) "%s: " fmt, __func__
|
|
|
|
#include <linux/err.h>
|
|
#include <linux/rtmutex.h>
|
|
#include <linux/clk.h>
|
|
#include <linux/clk/msm-clk-provider.h>
|
|
#include <soc/qcom/clock-voter.h>
|
|
#include <soc/qcom/msm-clock-controller.h>
|
|
|
|
static DEFINE_RT_MUTEX(voter_clk_lock);
|
|
|
|
/* Aggregate the rate of clocks that are currently on. */
|
|
static unsigned long voter_clk_aggregate_rate(const struct clk *parent)
|
|
{
|
|
struct clk *clk;
|
|
unsigned long rate = 0;
|
|
|
|
list_for_each_entry(clk, &parent->children, siblings) {
|
|
struct clk_voter *v = to_clk_voter(clk);
|
|
|
|
if (v->enabled)
|
|
rate = max(clk->rate, rate);
|
|
}
|
|
return rate;
|
|
}
|
|
|
|
static int voter_clk_set_rate(struct clk *clk, unsigned long rate)
|
|
{
|
|
int ret = 0;
|
|
struct clk *clkp;
|
|
struct clk_voter *clkh, *v = to_clk_voter(clk);
|
|
unsigned long cur_rate, new_rate, other_rate = 0;
|
|
|
|
if (v->is_branch)
|
|
return 0;
|
|
|
|
rt_mutex_lock(&voter_clk_lock);
|
|
|
|
if (v->enabled) {
|
|
struct clk *parent = clk->parent;
|
|
|
|
/*
|
|
* Get the aggregate rate without this clock's vote and update
|
|
* if the new rate is different than the current rate
|
|
*/
|
|
list_for_each_entry(clkp, &parent->children, siblings) {
|
|
clkh = to_clk_voter(clkp);
|
|
if (clkh->enabled && clkh != v)
|
|
other_rate = max(clkp->rate, other_rate);
|
|
}
|
|
|
|
cur_rate = max(other_rate, clk->rate);
|
|
new_rate = max(other_rate, rate);
|
|
|
|
if (new_rate != cur_rate) {
|
|
ret = clk_set_rate(parent, new_rate);
|
|
if (ret)
|
|
goto unlock;
|
|
}
|
|
}
|
|
clk->rate = rate;
|
|
unlock:
|
|
rt_mutex_unlock(&voter_clk_lock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int voter_clk_prepare(struct clk *clk)
|
|
{
|
|
int ret = 0;
|
|
unsigned long cur_rate;
|
|
struct clk *parent;
|
|
struct clk_voter *v = to_clk_voter(clk);
|
|
|
|
rt_mutex_lock(&voter_clk_lock);
|
|
parent = clk->parent;
|
|
|
|
if (v->is_branch) {
|
|
v->enabled = true;
|
|
goto out;
|
|
}
|
|
|
|
/*
|
|
* Increase the rate if this clock is voting for a higher rate
|
|
* than the current rate.
|
|
*/
|
|
cur_rate = voter_clk_aggregate_rate(parent);
|
|
if (clk->rate > cur_rate) {
|
|
ret = clk_set_rate(parent, clk->rate);
|
|
if (ret)
|
|
goto out;
|
|
}
|
|
v->enabled = true;
|
|
out:
|
|
rt_mutex_unlock(&voter_clk_lock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void voter_clk_unprepare(struct clk *clk)
|
|
{
|
|
unsigned long cur_rate, new_rate;
|
|
struct clk *parent;
|
|
struct clk_voter *v = to_clk_voter(clk);
|
|
|
|
|
|
rt_mutex_lock(&voter_clk_lock);
|
|
parent = clk->parent;
|
|
|
|
/*
|
|
* Decrease the rate if this clock was the only one voting for
|
|
* the highest rate.
|
|
*/
|
|
v->enabled = false;
|
|
if (v->is_branch)
|
|
goto out;
|
|
|
|
new_rate = voter_clk_aggregate_rate(parent);
|
|
cur_rate = max(new_rate, clk->rate);
|
|
|
|
if (new_rate < cur_rate)
|
|
clk_set_rate(parent, new_rate);
|
|
|
|
out:
|
|
rt_mutex_unlock(&voter_clk_lock);
|
|
}
|
|
|
|
static int voter_clk_is_enabled(struct clk *clk)
|
|
{
|
|
struct clk_voter *v = to_clk_voter(clk);
|
|
|
|
return v->enabled;
|
|
}
|
|
|
|
static long voter_clk_round_rate(struct clk *clk, unsigned long rate)
|
|
{
|
|
return clk_round_rate(clk->parent, rate);
|
|
}
|
|
|
|
static bool voter_clk_is_local(struct clk *clk)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
static enum handoff voter_clk_handoff(struct clk *clk)
|
|
{
|
|
if (!clk->rate)
|
|
return HANDOFF_DISABLED_CLK;
|
|
|
|
/*
|
|
* Send the default rate to the parent if necessary and update the
|
|
* software state of the voter clock.
|
|
*/
|
|
if (voter_clk_prepare(clk) < 0)
|
|
return HANDOFF_DISABLED_CLK;
|
|
|
|
return HANDOFF_ENABLED_CLK;
|
|
}
|
|
|
|
const struct clk_ops clk_ops_voter = {
|
|
.prepare = voter_clk_prepare,
|
|
.unprepare = voter_clk_unprepare,
|
|
.set_rate = voter_clk_set_rate,
|
|
.is_enabled = voter_clk_is_enabled,
|
|
.round_rate = voter_clk_round_rate,
|
|
.is_local = voter_clk_is_local,
|
|
.handoff = voter_clk_handoff,
|
|
};
|
|
|
|
static void *sw_vote_clk_dt_parser(struct device *dev,
|
|
struct device_node *np)
|
|
{
|
|
struct clk_voter *v;
|
|
int rc;
|
|
u32 temp;
|
|
|
|
v = devm_kzalloc(dev, sizeof(*v), GFP_KERNEL);
|
|
if (!v)
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
rc = of_property_read_u32(np, "qcom,config-rate", &temp);
|
|
if (rc) {
|
|
dt_prop_err(np, "qcom,config-rate", "is missing");
|
|
return ERR_PTR(rc);
|
|
}
|
|
|
|
v->c.ops = &clk_ops_voter;
|
|
return msmclk_generic_clk_init(dev, np, &v->c);
|
|
}
|
|
MSMCLK_PARSER(sw_vote_clk_dt_parser, "qcom,sw-vote-clk", 0);
|
|
|