Commit 4760a896 authored by Rajendra Nayak's avatar Rajendra Nayak Committed by Bjorn Andersson

remoteproc: q6v5-mss: Vote for rpmh power domains

With rpmh ARC resources being modelled as power domains with performance
state, we need to proxy vote on these for SDM845.
Add support to vote on multiple of them, now that genpd supports
associating mutliple power domains to a device.
Tested-by: default avatarSibi Sankar <sibis@codeaurora.org>
Reviewed-by: default avatarSibi Sankar <sibis@codeaurora.org>
Signed-off-by: default avatarRajendra Nayak <rnayak@codeaurora.org>
[bjorn: Drop device link, improve error handling, name things "proxy"]
Signed-off-by: default avatarBjorn Andersson <bjorn.andersson@linaro.org>
parent a5a4e02d
...@@ -25,6 +25,8 @@ ...@@ -25,6 +25,8 @@
#include <linux/of_address.h> #include <linux/of_address.h>
#include <linux/of_device.h> #include <linux/of_device.h>
#include <linux/platform_device.h> #include <linux/platform_device.h>
#include <linux/pm_domain.h>
#include <linux/pm_runtime.h>
#include <linux/regmap.h> #include <linux/regmap.h>
#include <linux/regulator/consumer.h> #include <linux/regulator/consumer.h>
#include <linux/remoteproc.h> #include <linux/remoteproc.h>
...@@ -131,6 +133,7 @@ struct rproc_hexagon_res { ...@@ -131,6 +133,7 @@ struct rproc_hexagon_res {
char **proxy_clk_names; char **proxy_clk_names;
char **reset_clk_names; char **reset_clk_names;
char **active_clk_names; char **active_clk_names;
char **proxy_pd_names;
int version; int version;
bool need_mem_protection; bool need_mem_protection;
bool has_alt_reset; bool has_alt_reset;
...@@ -156,9 +159,11 @@ struct q6v5 { ...@@ -156,9 +159,11 @@ struct q6v5 {
struct clk *active_clks[8]; struct clk *active_clks[8];
struct clk *reset_clks[4]; struct clk *reset_clks[4];
struct clk *proxy_clks[4]; struct clk *proxy_clks[4];
struct device *proxy_pds[3];
int active_clk_count; int active_clk_count;
int reset_clk_count; int reset_clk_count;
int proxy_clk_count; int proxy_clk_count;
int proxy_pd_count;
struct reg_info active_regs[1]; struct reg_info active_regs[1];
struct reg_info proxy_regs[3]; struct reg_info proxy_regs[3];
...@@ -322,6 +327,41 @@ static void q6v5_clk_disable(struct device *dev, ...@@ -322,6 +327,41 @@ static void q6v5_clk_disable(struct device *dev,
clk_disable_unprepare(clks[i]); clk_disable_unprepare(clks[i]);
} }
static int q6v5_pds_enable(struct q6v5 *qproc, struct device **pds,
size_t pd_count)
{
int ret;
int i;
for (i = 0; i < pd_count; i++) {
dev_pm_genpd_set_performance_state(pds[i], INT_MAX);
ret = pm_runtime_get_sync(pds[i]);
if (ret < 0)
goto unroll_pd_votes;
}
return 0;
unroll_pd_votes:
for (i--; i >= 0; i--) {
dev_pm_genpd_set_performance_state(pds[i], 0);
pm_runtime_put(pds[i]);
}
return ret;
};
static void q6v5_pds_disable(struct q6v5 *qproc, struct device **pds,
size_t pd_count)
{
int i;
for (i = 0; i < pd_count; i++) {
dev_pm_genpd_set_performance_state(pds[i], 0);
pm_runtime_put(pds[i]);
}
}
static int q6v5_xfer_mem_ownership(struct q6v5 *qproc, int *current_perm, static int q6v5_xfer_mem_ownership(struct q6v5 *qproc, int *current_perm,
bool remote_owner, phys_addr_t addr, bool remote_owner, phys_addr_t addr,
size_t size) size_t size)
...@@ -691,11 +731,17 @@ static int q6v5_mba_load(struct q6v5 *qproc) ...@@ -691,11 +731,17 @@ static int q6v5_mba_load(struct q6v5 *qproc)
qcom_q6v5_prepare(&qproc->q6v5); qcom_q6v5_prepare(&qproc->q6v5);
ret = q6v5_pds_enable(qproc, qproc->proxy_pds, qproc->proxy_pd_count);
if (ret < 0) {
dev_err(qproc->dev, "failed to enable proxy power domains\n");
goto disable_irqs;
}
ret = q6v5_regulator_enable(qproc, qproc->proxy_regs, ret = q6v5_regulator_enable(qproc, qproc->proxy_regs,
qproc->proxy_reg_count); qproc->proxy_reg_count);
if (ret) { if (ret) {
dev_err(qproc->dev, "failed to enable proxy supplies\n"); dev_err(qproc->dev, "failed to enable proxy supplies\n");
goto disable_irqs; goto disable_proxy_pds;
} }
ret = q6v5_clk_enable(qproc->dev, qproc->proxy_clks, ret = q6v5_clk_enable(qproc->dev, qproc->proxy_clks,
...@@ -792,6 +838,8 @@ static int q6v5_mba_load(struct q6v5 *qproc) ...@@ -792,6 +838,8 @@ static int q6v5_mba_load(struct q6v5 *qproc)
disable_proxy_reg: disable_proxy_reg:
q6v5_regulator_disable(qproc, qproc->proxy_regs, q6v5_regulator_disable(qproc, qproc->proxy_regs,
qproc->proxy_reg_count); qproc->proxy_reg_count);
disable_proxy_pds:
q6v5_pds_disable(qproc, qproc->proxy_pds, qproc->proxy_pd_count);
disable_irqs: disable_irqs:
qcom_q6v5_unprepare(&qproc->q6v5); qcom_q6v5_unprepare(&qproc->q6v5);
...@@ -842,6 +890,8 @@ static void q6v5_mba_reclaim(struct q6v5 *qproc) ...@@ -842,6 +890,8 @@ static void q6v5_mba_reclaim(struct q6v5 *qproc)
ret = qcom_q6v5_unprepare(&qproc->q6v5); ret = qcom_q6v5_unprepare(&qproc->q6v5);
if (ret) { if (ret) {
q6v5_pds_disable(qproc, qproc->proxy_pds,
qproc->proxy_pd_count);
q6v5_clk_disable(qproc->dev, qproc->proxy_clks, q6v5_clk_disable(qproc->dev, qproc->proxy_clks,
qproc->proxy_clk_count); qproc->proxy_clk_count);
q6v5_regulator_disable(qproc, qproc->proxy_regs, q6v5_regulator_disable(qproc, qproc->proxy_regs,
...@@ -1135,6 +1185,7 @@ static void qcom_msa_handover(struct qcom_q6v5 *q6v5) ...@@ -1135,6 +1185,7 @@ static void qcom_msa_handover(struct qcom_q6v5 *q6v5)
qproc->proxy_clk_count); qproc->proxy_clk_count);
q6v5_regulator_disable(qproc, qproc->proxy_regs, q6v5_regulator_disable(qproc, qproc->proxy_regs,
qproc->proxy_reg_count); qproc->proxy_reg_count);
q6v5_pds_disable(qproc, qproc->proxy_pds, qproc->proxy_pd_count);
} }
static int q6v5_init_mem(struct q6v5 *qproc, struct platform_device *pdev) static int q6v5_init_mem(struct q6v5 *qproc, struct platform_device *pdev)
...@@ -1195,6 +1246,45 @@ static int q6v5_init_clocks(struct device *dev, struct clk **clks, ...@@ -1195,6 +1246,45 @@ static int q6v5_init_clocks(struct device *dev, struct clk **clks,
return i; return i;
} }
static int q6v5_pds_attach(struct device *dev, struct device **devs,
char **pd_names)
{
size_t num_pds = 0;
int ret;
int i;
if (!pd_names)
return 0;
while (pd_names[num_pds])
num_pds++;
for (i = 0; i < num_pds; i++) {
devs[i] = dev_pm_domain_attach_by_name(dev, pd_names[i]);
if (IS_ERR(devs[i])) {
ret = PTR_ERR(devs[i]);
goto unroll_attach;
}
}
return num_pds;
unroll_attach:
for (i--; i >= 0; i--)
dev_pm_domain_detach(devs[i], false);
return ret;
};
static void q6v5_pds_detach(struct q6v5 *qproc, struct device **pds,
size_t pd_count)
{
int i;
for (i = 0; i < pd_count; i++)
dev_pm_domain_detach(pds[i], false);
}
static int q6v5_init_reset(struct q6v5 *qproc) static int q6v5_init_reset(struct q6v5 *qproc)
{ {
qproc->mss_restart = devm_reset_control_get_exclusive(qproc->dev, qproc->mss_restart = devm_reset_control_get_exclusive(qproc->dev,
...@@ -1351,10 +1441,18 @@ static int q6v5_probe(struct platform_device *pdev) ...@@ -1351,10 +1441,18 @@ static int q6v5_probe(struct platform_device *pdev)
} }
qproc->active_reg_count = ret; qproc->active_reg_count = ret;
ret = q6v5_pds_attach(&pdev->dev, qproc->proxy_pds,
desc->proxy_pd_names);
if (ret < 0) {
dev_err(&pdev->dev, "Failed to init power domains\n");
goto free_rproc;
}
qproc->proxy_pd_count = ret;
qproc->has_alt_reset = desc->has_alt_reset; qproc->has_alt_reset = desc->has_alt_reset;
ret = q6v5_init_reset(qproc); ret = q6v5_init_reset(qproc);
if (ret) if (ret)
goto free_rproc; goto detach_proxy_pds;
qproc->version = desc->version; qproc->version = desc->version;
qproc->need_mem_protection = desc->need_mem_protection; qproc->need_mem_protection = desc->need_mem_protection;
...@@ -1362,7 +1460,7 @@ static int q6v5_probe(struct platform_device *pdev) ...@@ -1362,7 +1460,7 @@ static int q6v5_probe(struct platform_device *pdev)
ret = qcom_q6v5_init(&qproc->q6v5, pdev, rproc, MPSS_CRASH_REASON_SMEM, ret = qcom_q6v5_init(&qproc->q6v5, pdev, rproc, MPSS_CRASH_REASON_SMEM,
qcom_msa_handover); qcom_msa_handover);
if (ret) if (ret)
goto free_rproc; goto detach_proxy_pds;
qproc->mpss_perm = BIT(QCOM_SCM_VMID_HLOS); qproc->mpss_perm = BIT(QCOM_SCM_VMID_HLOS);
qproc->mba_perm = BIT(QCOM_SCM_VMID_HLOS); qproc->mba_perm = BIT(QCOM_SCM_VMID_HLOS);
...@@ -1372,15 +1470,17 @@ static int q6v5_probe(struct platform_device *pdev) ...@@ -1372,15 +1470,17 @@ static int q6v5_probe(struct platform_device *pdev)
qproc->sysmon = qcom_add_sysmon_subdev(rproc, "modem", 0x12); qproc->sysmon = qcom_add_sysmon_subdev(rproc, "modem", 0x12);
if (IS_ERR(qproc->sysmon)) { if (IS_ERR(qproc->sysmon)) {
ret = PTR_ERR(qproc->sysmon); ret = PTR_ERR(qproc->sysmon);
goto free_rproc; goto detach_proxy_pds;
} }
ret = rproc_add(rproc); ret = rproc_add(rproc);
if (ret) if (ret)
goto free_rproc; goto detach_proxy_pds;
return 0; return 0;
detach_proxy_pds:
q6v5_pds_detach(qproc, qproc->proxy_pds, qproc->proxy_pd_count);
free_rproc: free_rproc:
rproc_free(rproc); rproc_free(rproc);
...@@ -1397,6 +1497,9 @@ static int q6v5_remove(struct platform_device *pdev) ...@@ -1397,6 +1497,9 @@ static int q6v5_remove(struct platform_device *pdev)
qcom_remove_glink_subdev(qproc->rproc, &qproc->glink_subdev); qcom_remove_glink_subdev(qproc->rproc, &qproc->glink_subdev);
qcom_remove_smd_subdev(qproc->rproc, &qproc->smd_subdev); qcom_remove_smd_subdev(qproc->rproc, &qproc->smd_subdev);
qcom_remove_ssr_subdev(qproc->rproc, &qproc->ssr_subdev); qcom_remove_ssr_subdev(qproc->rproc, &qproc->ssr_subdev);
q6v5_pds_detach(qproc, qproc->proxy_pds, qproc->proxy_pd_count);
rproc_free(qproc->rproc); rproc_free(qproc->rproc);
return 0; return 0;
...@@ -1421,6 +1524,12 @@ static const struct rproc_hexagon_res sdm845_mss = { ...@@ -1421,6 +1524,12 @@ static const struct rproc_hexagon_res sdm845_mss = {
"mnoc_axi", "mnoc_axi",
NULL NULL
}, },
.proxy_pd_names = (char*[]){
"cx",
"mx",
"mss",
NULL
},
.need_mem_protection = true, .need_mem_protection = true,
.has_alt_reset = true, .has_alt_reset = true,
.version = MSS_SDM845, .version = MSS_SDM845,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment