1037 lines
24 KiB
C
1037 lines
24 KiB
C
/* Copyright (c) 2014-2015, 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.
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/err.h>
|
|
#include <linux/io.h>
|
|
#include <linux/module.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/of.h>
|
|
#include <linux/clk.h>
|
|
#include <linux/regulator/consumer.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/of_gpio.h>
|
|
#include <linux/delay.h>
|
|
|
|
#include <linux/msm-bus-board.h>
|
|
#include <linux/msm-bus.h>
|
|
#include <linux/dma-mapping.h>
|
|
|
|
#include <soc/qcom/subsystem_restart.h>
|
|
#include <soc/qcom/ramdump.h>
|
|
#include <soc/qcom/scm.h>
|
|
|
|
#include <soc/qcom/smem.h>
|
|
|
|
#include "peripheral-loader.h"
|
|
|
|
#define XO_FREQ 19200000
|
|
#define PROXY_TIMEOUT_MS 10000
|
|
#define MAX_SSR_REASON_LEN 81U
|
|
#define STOP_ACK_TIMEOUT_MS 1000
|
|
#define CRASH_STOP_ACK_TO_MS 200
|
|
|
|
#define desc_to_data(d) container_of(d, struct pil_tz_data, desc)
|
|
#define subsys_to_data(d) container_of(d, struct pil_tz_data, subsys_desc)
|
|
|
|
/**
|
|
* struct reg_info - regulator info
|
|
* @reg: regulator handle
|
|
* @uV: voltage in uV
|
|
* @uA: current in uA
|
|
*/
|
|
struct reg_info {
|
|
struct regulator *reg;
|
|
int uV;
|
|
int uA;
|
|
};
|
|
|
|
/**
|
|
* struct pil_tz_data
|
|
* @regs: regulators that should be always on when the subsystem is
|
|
* brought out of reset
|
|
* @proxy_regs: regulators that should be on during pil proxy voting
|
|
* @clks: clocks that should be always on when the subsystem is
|
|
* brought out of reset
|
|
* @proxy_clks: clocks that should be on during pil proxy voting
|
|
* @reg_count: the number of always on regulators
|
|
* @proxy_reg_count: the number of proxy voting regulators
|
|
* @clk_count: the number of always on clocks
|
|
* @proxy_clk_count: the number of proxy voting clocks
|
|
* @smem_id: the smem id used for read the subsystem crash reason
|
|
* @ramdump_dev: ramdump device pointer
|
|
* @pas_id: the PAS id for tz
|
|
* @bus_client: bus client id
|
|
* @enable_bus_scaling: set to true if PIL needs to vote for
|
|
* bus bandwidth
|
|
* @keep_proxy_regs_on: If set, during proxy unvoting, PIL removes the
|
|
* voltage/current vote for proxy regulators but leaves
|
|
* them enabled.
|
|
* @stop_ack: state of completion of stop ack
|
|
* @desc: PIL descriptor
|
|
* @subsys: subsystem device pointer
|
|
* @subsys_desc: subsystem descriptor
|
|
*/
|
|
struct pil_tz_data {
|
|
struct reg_info *regs;
|
|
struct reg_info *proxy_regs;
|
|
struct clk **clks;
|
|
struct clk **proxy_clks;
|
|
int reg_count;
|
|
int proxy_reg_count;
|
|
int clk_count;
|
|
int proxy_clk_count;
|
|
int smem_id;
|
|
void *ramdump_dev;
|
|
u32 pas_id;
|
|
u32 bus_client;
|
|
bool enable_bus_scaling;
|
|
bool keep_proxy_regs_on;
|
|
struct completion stop_ack;
|
|
struct pil_desc desc;
|
|
struct subsys_device *subsys;
|
|
struct subsys_desc subsys_desc;
|
|
};
|
|
|
|
enum scm_cmd {
|
|
PAS_INIT_IMAGE_CMD = 1,
|
|
PAS_MEM_SETUP_CMD,
|
|
PAS_AUTH_AND_RESET_CMD = 5,
|
|
PAS_SHUTDOWN_CMD,
|
|
};
|
|
|
|
enum pas_id {
|
|
PAS_MODEM,
|
|
PAS_Q6,
|
|
PAS_DSPS,
|
|
PAS_TZAPPS,
|
|
PAS_MODEM_SW,
|
|
PAS_MODEM_FW,
|
|
PAS_WCNSS,
|
|
PAS_SECAPP,
|
|
PAS_GSS,
|
|
PAS_VIDC,
|
|
PAS_VPU,
|
|
PAS_BCSS,
|
|
};
|
|
|
|
static struct msm_bus_paths scm_pas_bw_tbl[] = {
|
|
{
|
|
.vectors = (struct msm_bus_vectors[]){
|
|
{
|
|
.src = MSM_BUS_MASTER_SPS,
|
|
.dst = MSM_BUS_SLAVE_EBI_CH0,
|
|
},
|
|
},
|
|
.num_paths = 1,
|
|
},
|
|
{
|
|
.vectors = (struct msm_bus_vectors[]){
|
|
{
|
|
.src = MSM_BUS_MASTER_SPS,
|
|
.dst = MSM_BUS_SLAVE_EBI_CH0,
|
|
.ib = 492 * 8 * 1000000UL,
|
|
.ab = 492 * 8 * 100000UL,
|
|
},
|
|
},
|
|
.num_paths = 1,
|
|
},
|
|
};
|
|
|
|
static struct msm_bus_scale_pdata scm_pas_bus_pdata = {
|
|
.usecase = scm_pas_bw_tbl,
|
|
.num_usecases = ARRAY_SIZE(scm_pas_bw_tbl),
|
|
.name = "scm_pas",
|
|
};
|
|
|
|
static uint32_t scm_perf_client;
|
|
static int scm_pas_bw_count;
|
|
static DEFINE_MUTEX(scm_pas_bw_mutex);
|
|
|
|
static int scm_pas_enable_bw(void)
|
|
{
|
|
int ret = 0;
|
|
|
|
if (!scm_perf_client)
|
|
return -EINVAL;
|
|
|
|
mutex_lock(&scm_pas_bw_mutex);
|
|
if (!scm_pas_bw_count) {
|
|
ret = msm_bus_scale_client_update_request(scm_perf_client, 1);
|
|
if (ret)
|
|
goto err_bus;
|
|
scm_pas_bw_count++;
|
|
}
|
|
|
|
mutex_unlock(&scm_pas_bw_mutex);
|
|
return ret;
|
|
|
|
err_bus:
|
|
pr_err("scm-pas; Bandwidth request failed (%d)\n", ret);
|
|
msm_bus_scale_client_update_request(scm_perf_client, 0);
|
|
|
|
mutex_unlock(&scm_pas_bw_mutex);
|
|
return ret;
|
|
}
|
|
|
|
static void scm_pas_disable_bw(void)
|
|
{
|
|
mutex_lock(&scm_pas_bw_mutex);
|
|
if (scm_pas_bw_count-- == 1)
|
|
msm_bus_scale_client_update_request(scm_perf_client, 0);
|
|
mutex_unlock(&scm_pas_bw_mutex);
|
|
}
|
|
|
|
static void scm_pas_init(int id)
|
|
{
|
|
static int is_inited;
|
|
|
|
if (is_inited)
|
|
return;
|
|
|
|
scm_pas_bw_tbl[0].vectors[0].src = id;
|
|
scm_pas_bw_tbl[1].vectors[0].src = id;
|
|
|
|
scm_perf_client = msm_bus_scale_register_client(&scm_pas_bus_pdata);
|
|
if (!scm_perf_client)
|
|
pr_warn("scm-pas: Unable to register bus client\n");
|
|
|
|
is_inited = 1;
|
|
}
|
|
|
|
static int of_read_clocks(struct device *dev, struct clk ***clks_ref,
|
|
const char *propname)
|
|
{
|
|
int clk_count, i, len;
|
|
struct clk **clks;
|
|
|
|
if (!of_find_property(dev->of_node, propname, &len))
|
|
return 0;
|
|
|
|
clk_count = of_property_count_strings(dev->of_node, propname);
|
|
if (IS_ERR_VALUE(clk_count)) {
|
|
dev_err(dev, "Failed to get clock names\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
clks = devm_kzalloc(dev, sizeof(struct clk *) * clk_count,
|
|
GFP_KERNEL);
|
|
if (!clks)
|
|
return -ENOMEM;
|
|
|
|
for (i = 0; i < clk_count; i++) {
|
|
const char *clock_name;
|
|
char clock_freq_name[50];
|
|
u32 clock_rate = XO_FREQ;
|
|
|
|
of_property_read_string_index(dev->of_node,
|
|
propname, i,
|
|
&clock_name);
|
|
snprintf(clock_freq_name, ARRAY_SIZE(clock_freq_name),
|
|
"qcom,%s-freq", clock_name);
|
|
if (of_find_property(dev->of_node, clock_freq_name, &len))
|
|
if (of_property_read_u32(dev->of_node, clock_freq_name,
|
|
&clock_rate)) {
|
|
dev_err(dev, "Failed to read %s clock's freq\n",
|
|
clock_freq_name);
|
|
return -EINVAL;
|
|
}
|
|
|
|
clks[i] = devm_clk_get(dev, clock_name);
|
|
if (IS_ERR(clks[i])) {
|
|
int rc = PTR_ERR(clks[i]);
|
|
if (rc != -EPROBE_DEFER)
|
|
dev_err(dev, "Failed to get %s clock\n",
|
|
clock_name);
|
|
return rc;
|
|
}
|
|
|
|
/* Make sure rate-settable clocks' rates are set */
|
|
if (clk_get_rate(clks[i]) == 0)
|
|
clk_set_rate(clks[i], clk_round_rate(clks[i],
|
|
clock_rate));
|
|
}
|
|
|
|
*clks_ref = clks;
|
|
return clk_count;
|
|
}
|
|
|
|
static int of_read_regs(struct device *dev, struct reg_info **regs_ref,
|
|
const char *propname)
|
|
{
|
|
int reg_count, i, len, rc;
|
|
struct reg_info *regs;
|
|
|
|
if (!of_find_property(dev->of_node, propname, &len))
|
|
return 0;
|
|
|
|
reg_count = of_property_count_strings(dev->of_node, propname);
|
|
if (IS_ERR_VALUE(reg_count)) {
|
|
dev_err(dev, "Failed to get regulator names\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
regs = devm_kzalloc(dev, sizeof(struct reg_info) * reg_count,
|
|
GFP_KERNEL);
|
|
if (!regs)
|
|
return -ENOMEM;
|
|
|
|
for (i = 0; i < reg_count; i++) {
|
|
const char *reg_name;
|
|
char reg_uV_uA_name[50];
|
|
u32 vdd_uV_uA[2];
|
|
|
|
of_property_read_string_index(dev->of_node,
|
|
propname, i,
|
|
®_name);
|
|
|
|
regs[i].reg = devm_regulator_get(dev, reg_name);
|
|
if (IS_ERR(regs[i].reg)) {
|
|
int rc = PTR_ERR(regs[i].reg);
|
|
if (rc != -EPROBE_DEFER)
|
|
dev_err(dev, "Failed to get %s\n regulator",
|
|
reg_name);
|
|
return rc;
|
|
}
|
|
|
|
/*
|
|
* Read the voltage and current values for the corresponding
|
|
* regulator. The device tree property name is "qcom," +
|
|
* "regulator_name" + "-uV-uA".
|
|
*/
|
|
rc = snprintf(reg_uV_uA_name, ARRAY_SIZE(reg_uV_uA_name),
|
|
"qcom,%s-uV-uA", reg_name);
|
|
if (rc < strlen(reg_name) + 6) {
|
|
dev_err(dev, "Failed to hold reg_uV_uA_name\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!of_find_property(dev->of_node, reg_uV_uA_name, &len))
|
|
continue;
|
|
|
|
len /= sizeof(vdd_uV_uA[0]);
|
|
|
|
/* There should be two entries: one for uV and one for uA */
|
|
if (len != 2) {
|
|
dev_err(dev, "Missing uV/uA value\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
rc = of_property_read_u32_array(dev->of_node, reg_uV_uA_name,
|
|
vdd_uV_uA, len);
|
|
if (rc) {
|
|
dev_err(dev, "Failed to read uV/uA values\n");
|
|
return rc;
|
|
}
|
|
|
|
regs[i].uV = vdd_uV_uA[0];
|
|
regs[i].uA = vdd_uV_uA[1];
|
|
}
|
|
|
|
*regs_ref = regs;
|
|
return reg_count;
|
|
}
|
|
|
|
static int of_read_bus_pdata(struct platform_device *pdev,
|
|
struct pil_tz_data *d)
|
|
{
|
|
struct msm_bus_scale_pdata *pdata;
|
|
pdata = msm_bus_cl_get_pdata(pdev);
|
|
|
|
if (!pdata)
|
|
return -EINVAL;
|
|
|
|
d->bus_client = msm_bus_scale_register_client(pdata);
|
|
if (!d->bus_client)
|
|
pr_warn("%s: Unable to register bus client\n", __func__);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int piltz_resc_init(struct platform_device *pdev, struct pil_tz_data *d)
|
|
{
|
|
int len, count, rc;
|
|
struct device *dev = &pdev->dev;
|
|
|
|
count = of_read_clocks(dev, &d->clks, "qcom,active-clock-names");
|
|
if (count < 0) {
|
|
dev_err(dev, "Failed to setup clocks.\n");
|
|
return count;
|
|
}
|
|
d->clk_count = count;
|
|
|
|
count = of_read_clocks(dev, &d->proxy_clks, "qcom,proxy-clock-names");
|
|
if (count < 0) {
|
|
dev_err(dev, "Failed to setup proxy clocks.\n");
|
|
return count;
|
|
}
|
|
d->proxy_clk_count = count;
|
|
|
|
count = of_read_regs(dev, &d->regs, "qcom,active-reg-names");
|
|
if (count < 0) {
|
|
dev_err(dev, "Failed to setup regulators.\n");
|
|
return count;
|
|
}
|
|
d->reg_count = count;
|
|
|
|
count = of_read_regs(dev, &d->proxy_regs, "qcom,proxy-reg-names");
|
|
if (count < 0) {
|
|
dev_err(dev, "Failed to setup proxy regulators.\n");
|
|
return count;
|
|
}
|
|
d->proxy_reg_count = count;
|
|
|
|
if (of_find_property(dev->of_node, "qcom,msm-bus,name", &len)) {
|
|
d->enable_bus_scaling = true;
|
|
rc = of_read_bus_pdata(pdev, d);
|
|
if (rc) {
|
|
dev_err(dev, "Failed to setup bus scaling client.\n");
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int enable_regulators(struct pil_tz_data *d, struct device *dev,
|
|
struct reg_info *regs, int reg_count,
|
|
bool reg_no_enable)
|
|
{
|
|
int i, rc = 0;
|
|
|
|
for (i = 0; i < reg_count; i++) {
|
|
if (regs[i].uV > 0) {
|
|
rc = regulator_set_voltage(regs[i].reg,
|
|
regs[i].uV, INT_MAX);
|
|
if (rc) {
|
|
dev_err(dev, "Failed to request voltage.\n");
|
|
goto err_voltage;
|
|
}
|
|
}
|
|
|
|
if (regs[i].uA > 0) {
|
|
rc = regulator_set_optimum_mode(regs[i].reg,
|
|
regs[i].uA);
|
|
if (rc < 0) {
|
|
dev_err(dev, "Failed to set regulator mode\n");
|
|
goto err_mode;
|
|
}
|
|
}
|
|
|
|
if (d->keep_proxy_regs_on && reg_no_enable)
|
|
continue;
|
|
|
|
rc = regulator_enable(regs[i].reg);
|
|
if (rc) {
|
|
dev_err(dev, "Regulator enable failed\n");
|
|
goto err_enable;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
err_enable:
|
|
if (regs[i].uA > 0) {
|
|
regulator_set_voltage(regs[i].reg, 0, INT_MAX);
|
|
regulator_set_optimum_mode(regs[i].reg, 0);
|
|
}
|
|
err_mode:
|
|
if (regs[i].uV > 0)
|
|
regulator_set_voltage(regs[i].reg, 0, INT_MAX);
|
|
err_voltage:
|
|
for (i--; i >= 0; i--) {
|
|
if (regs[i].uV > 0)
|
|
regulator_set_voltage(regs[i].reg, 0, INT_MAX);
|
|
|
|
if (regs[i].uA > 0)
|
|
regulator_set_optimum_mode(regs[i].reg, 0);
|
|
|
|
if (d->keep_proxy_regs_on && reg_no_enable)
|
|
continue;
|
|
regulator_disable(regs[i].reg);
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
static void disable_regulators(struct pil_tz_data *d, struct reg_info *regs,
|
|
int reg_count, bool reg_no_disable)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < reg_count; i++) {
|
|
if (regs[i].uV > 0)
|
|
regulator_set_voltage(regs[i].reg, 0, INT_MAX);
|
|
|
|
if (regs[i].uA > 0)
|
|
regulator_set_optimum_mode(regs[i].reg, 0);
|
|
|
|
if (d->keep_proxy_regs_on && reg_no_disable)
|
|
continue;
|
|
regulator_disable(regs[i].reg);
|
|
}
|
|
}
|
|
|
|
static int prepare_enable_clocks(struct device *dev, struct clk **clks,
|
|
int clk_count)
|
|
{
|
|
int rc = 0;
|
|
int i;
|
|
|
|
for (i = 0; i < clk_count; i++) {
|
|
rc = clk_prepare_enable(clks[i]);
|
|
if (rc) {
|
|
dev_err(dev, "Clock enable failed\n");
|
|
goto err;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
err:
|
|
for (i--; i >= 0; i--)
|
|
clk_disable_unprepare(clks[i]);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static void disable_unprepare_clocks(struct clk **clks, int clk_count)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < clk_count; i++)
|
|
clk_disable_unprepare(clks[i]);
|
|
}
|
|
|
|
static int pil_make_proxy_vote(struct pil_desc *pil)
|
|
{
|
|
struct pil_tz_data *d = desc_to_data(pil);
|
|
int rc;
|
|
|
|
if (d->subsys_desc.no_auth)
|
|
return 0;
|
|
|
|
rc = enable_regulators(d, pil->dev, d->proxy_regs,
|
|
d->proxy_reg_count, false);
|
|
if (rc)
|
|
return rc;
|
|
|
|
rc = prepare_enable_clocks(pil->dev, d->proxy_clks,
|
|
d->proxy_clk_count);
|
|
if (rc)
|
|
goto err_clks;
|
|
|
|
if (d->bus_client) {
|
|
rc = msm_bus_scale_client_update_request(d->bus_client, 1);
|
|
if (rc) {
|
|
dev_err(pil->dev, "bandwidth request failed\n");
|
|
goto err_bw;
|
|
}
|
|
} else
|
|
WARN(d->enable_bus_scaling, "Bus scaling not set up for %s!\n",
|
|
d->subsys_desc.name);
|
|
|
|
return 0;
|
|
err_bw:
|
|
disable_unprepare_clocks(d->proxy_clks, d->proxy_clk_count);
|
|
err_clks:
|
|
disable_regulators(d, d->proxy_regs, d->proxy_reg_count, false);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static void pil_remove_proxy_vote(struct pil_desc *pil)
|
|
{
|
|
struct pil_tz_data *d = desc_to_data(pil);
|
|
|
|
if (d->subsys_desc.no_auth)
|
|
return;
|
|
|
|
if (d->bus_client)
|
|
msm_bus_scale_client_update_request(d->bus_client, 0);
|
|
else
|
|
WARN(d->enable_bus_scaling, "Bus scaling not set up for %s!\n",
|
|
d->subsys_desc.name);
|
|
|
|
disable_unprepare_clocks(d->proxy_clks, d->proxy_clk_count);
|
|
|
|
disable_regulators(d, d->proxy_regs, d->proxy_reg_count, true);
|
|
}
|
|
|
|
static int pil_init_image_trusted(struct pil_desc *pil,
|
|
const u8 *metadata, size_t size)
|
|
{
|
|
struct pil_tz_data *d = desc_to_data(pil);
|
|
struct pas_init_image_req {
|
|
u32 proc;
|
|
u32 image_addr;
|
|
} request;
|
|
u32 scm_ret = 0;
|
|
void *mdata_buf;
|
|
dma_addr_t mdata_phys;
|
|
int ret;
|
|
DEFINE_DMA_ATTRS(attrs);
|
|
struct device dev = {0};
|
|
struct scm_desc desc = {0};
|
|
|
|
if (d->subsys_desc.no_auth)
|
|
return 0;
|
|
|
|
ret = scm_pas_enable_bw();
|
|
if (ret)
|
|
return ret;
|
|
dev.coherent_dma_mask =
|
|
DMA_BIT_MASK(sizeof(dma_addr_t) * 8);
|
|
dma_set_attr(DMA_ATTR_STRONGLY_ORDERED, &attrs);
|
|
mdata_buf = dma_alloc_attrs(&dev, size, &mdata_phys, GFP_KERNEL,
|
|
&attrs);
|
|
if (!mdata_buf) {
|
|
pr_err("scm-pas: Allocation for metadata failed.\n");
|
|
scm_pas_disable_bw();
|
|
return -ENOMEM;
|
|
}
|
|
|
|
memcpy(mdata_buf, metadata, size);
|
|
|
|
request.proc = d->pas_id;
|
|
request.image_addr = mdata_phys;
|
|
|
|
if (!is_scm_armv8()) {
|
|
ret = scm_call(SCM_SVC_PIL, PAS_INIT_IMAGE_CMD, &request,
|
|
sizeof(request), &scm_ret, sizeof(scm_ret));
|
|
} else {
|
|
desc.args[0] = d->pas_id;
|
|
desc.args[1] = mdata_phys;
|
|
desc.arginfo = SCM_ARGS(2, SCM_VAL, SCM_RW);
|
|
ret = scm_call2(SCM_SIP_FNID(SCM_SVC_PIL, PAS_INIT_IMAGE_CMD),
|
|
&desc);
|
|
scm_ret = desc.ret[0];
|
|
}
|
|
|
|
dma_free_attrs(&dev, size, mdata_buf, mdata_phys, &attrs);
|
|
scm_pas_disable_bw();
|
|
if (ret)
|
|
return ret;
|
|
return scm_ret;
|
|
}
|
|
|
|
static int pil_mem_setup_trusted(struct pil_desc *pil, phys_addr_t addr,
|
|
size_t size)
|
|
{
|
|
struct pil_tz_data *d = desc_to_data(pil);
|
|
struct pas_init_image_req {
|
|
u32 proc;
|
|
u32 start_addr;
|
|
u32 len;
|
|
} request;
|
|
u32 scm_ret = 0;
|
|
int ret;
|
|
struct scm_desc desc = {0};
|
|
|
|
if (d->subsys_desc.no_auth)
|
|
return 0;
|
|
|
|
request.proc = d->pas_id;
|
|
request.start_addr = addr;
|
|
request.len = size;
|
|
|
|
if (!is_scm_armv8()) {
|
|
ret = scm_call(SCM_SVC_PIL, PAS_MEM_SETUP_CMD, &request,
|
|
sizeof(request), &scm_ret, sizeof(scm_ret));
|
|
} else {
|
|
desc.args[0] = d->pas_id;
|
|
desc.args[1] = addr;
|
|
desc.args[2] = size;
|
|
desc.arginfo = SCM_ARGS(3);
|
|
ret = scm_call2(SCM_SIP_FNID(SCM_SVC_PIL, PAS_MEM_SETUP_CMD),
|
|
&desc);
|
|
scm_ret = desc.ret[0];
|
|
}
|
|
if (ret)
|
|
return ret;
|
|
return scm_ret;
|
|
}
|
|
|
|
static int pil_auth_and_reset(struct pil_desc *pil)
|
|
{
|
|
struct pil_tz_data *d = desc_to_data(pil);
|
|
int rc;
|
|
u32 proc, scm_ret = 0;
|
|
struct scm_desc desc = {0};
|
|
|
|
if (d->subsys_desc.no_auth)
|
|
return 0;
|
|
|
|
desc.args[0] = proc = d->pas_id;
|
|
desc.arginfo = SCM_ARGS(1);
|
|
|
|
rc = enable_regulators(d, pil->dev, d->regs, d->reg_count, false);
|
|
if (rc)
|
|
return rc;
|
|
|
|
rc = prepare_enable_clocks(pil->dev, d->clks, d->clk_count);
|
|
if (rc)
|
|
goto err_clks;
|
|
|
|
rc = scm_pas_enable_bw();
|
|
if (rc)
|
|
goto err_reset;
|
|
|
|
if (!is_scm_armv8()) {
|
|
rc = scm_call(SCM_SVC_PIL, PAS_AUTH_AND_RESET_CMD, &proc,
|
|
sizeof(proc), &scm_ret, sizeof(scm_ret));
|
|
} else {
|
|
rc = scm_call2(SCM_SIP_FNID(SCM_SVC_PIL,
|
|
PAS_AUTH_AND_RESET_CMD), &desc);
|
|
scm_ret = desc.ret[0];
|
|
}
|
|
scm_pas_disable_bw();
|
|
if (rc)
|
|
goto err_reset;
|
|
|
|
return scm_ret;
|
|
err_reset:
|
|
disable_unprepare_clocks(d->clks, d->clk_count);
|
|
err_clks:
|
|
disable_regulators(d, d->regs, d->reg_count, false);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int pil_shutdown_trusted(struct pil_desc *pil)
|
|
{
|
|
struct pil_tz_data *d = desc_to_data(pil);
|
|
u32 proc, scm_ret = 0;
|
|
int rc;
|
|
struct scm_desc desc = {0};
|
|
|
|
if (d->subsys_desc.no_auth)
|
|
return 0;
|
|
|
|
desc.args[0] = proc = d->pas_id;
|
|
desc.arginfo = SCM_ARGS(1);
|
|
|
|
rc = enable_regulators(d, pil->dev, d->proxy_regs,
|
|
d->proxy_reg_count, true);
|
|
if (rc)
|
|
return rc;
|
|
|
|
rc = prepare_enable_clocks(pil->dev, d->proxy_clks,
|
|
d->proxy_clk_count);
|
|
if (rc)
|
|
goto err_clks;
|
|
|
|
if (!is_scm_armv8()) {
|
|
rc = scm_call(SCM_SVC_PIL, PAS_SHUTDOWN_CMD, &proc,
|
|
sizeof(proc), &scm_ret, sizeof(scm_ret));
|
|
} else {
|
|
rc = scm_call2(SCM_SIP_FNID(SCM_SVC_PIL, PAS_SHUTDOWN_CMD),
|
|
&desc);
|
|
scm_ret = desc.ret[0];
|
|
}
|
|
|
|
disable_unprepare_clocks(d->proxy_clks, d->proxy_clk_count);
|
|
disable_regulators(d, d->proxy_regs, d->proxy_reg_count, false);
|
|
|
|
if (rc)
|
|
return rc;
|
|
|
|
disable_unprepare_clocks(d->clks, d->clk_count);
|
|
disable_regulators(d, d->regs, d->reg_count, false);
|
|
|
|
return scm_ret;
|
|
err_clks:
|
|
disable_regulators(d, d->proxy_regs, d->proxy_reg_count, false);
|
|
return rc;
|
|
}
|
|
|
|
static struct pil_reset_ops pil_ops_trusted = {
|
|
.init_image = pil_init_image_trusted,
|
|
.mem_setup = pil_mem_setup_trusted,
|
|
.auth_and_reset = pil_auth_and_reset,
|
|
.shutdown = pil_shutdown_trusted,
|
|
.proxy_vote = pil_make_proxy_vote,
|
|
.proxy_unvote = pil_remove_proxy_vote,
|
|
};
|
|
|
|
static void log_failure_reason(const struct pil_tz_data *d)
|
|
{
|
|
u32 size;
|
|
char *smem_reason, reason[MAX_SSR_REASON_LEN];
|
|
const char *name = d->subsys_desc.name;
|
|
|
|
if (d->smem_id == -1)
|
|
return;
|
|
|
|
smem_reason = smem_get_entry_no_rlock(d->smem_id, &size, 0,
|
|
SMEM_ANY_HOST_FLAG);
|
|
if (!smem_reason || !size) {
|
|
pr_err("%s SFR: (unknown, smem_get_entry_no_rlock failed).\n",
|
|
name);
|
|
return;
|
|
}
|
|
if (!smem_reason[0]) {
|
|
pr_err("%s SFR: (unknown, empty string found).\n", name);
|
|
return;
|
|
}
|
|
|
|
strlcpy(reason, smem_reason, min(size, MAX_SSR_REASON_LEN));
|
|
pr_err("%s subsystem failure reason: %s.\n", name, reason);
|
|
|
|
smem_reason[0] = '\0';
|
|
wmb();
|
|
}
|
|
|
|
static int subsys_shutdown(const struct subsys_desc *subsys, bool force_stop)
|
|
{
|
|
struct pil_tz_data *d = subsys_to_data(subsys);
|
|
int ret;
|
|
|
|
if (!subsys_get_crash_status(d->subsys) && force_stop &&
|
|
subsys->force_stop_gpio) {
|
|
gpio_set_value(subsys->force_stop_gpio, 1);
|
|
ret = wait_for_completion_timeout(&d->stop_ack,
|
|
msecs_to_jiffies(STOP_ACK_TIMEOUT_MS));
|
|
if (!ret)
|
|
pr_warn("Timed out on stop ack from %s.\n",
|
|
subsys->name);
|
|
gpio_set_value(subsys->force_stop_gpio, 0);
|
|
}
|
|
|
|
pil_shutdown(&d->desc);
|
|
return 0;
|
|
}
|
|
|
|
static int subsys_powerup(const struct subsys_desc *subsys)
|
|
{
|
|
struct pil_tz_data *d = subsys_to_data(subsys);
|
|
int ret = 0;
|
|
|
|
if (subsys->stop_ack_irq)
|
|
reinit_completion(&d->stop_ack);
|
|
|
|
d->desc.fw_name = subsys->fw_name;
|
|
ret = pil_boot(&d->desc);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int subsys_ramdump(int enable, const struct subsys_desc *subsys)
|
|
{
|
|
struct pil_tz_data *d = subsys_to_data(subsys);
|
|
|
|
if (!enable)
|
|
return 0;
|
|
|
|
return pil_do_ramdump(&d->desc, d->ramdump_dev);
|
|
}
|
|
|
|
static void subsys_free_memory(const struct subsys_desc *subsys)
|
|
{
|
|
struct pil_tz_data *d = subsys_to_data(subsys);
|
|
|
|
pil_free_memory(&d->desc);
|
|
}
|
|
|
|
static void subsys_crash_shutdown(const struct subsys_desc *subsys)
|
|
{
|
|
struct pil_tz_data *d = subsys_to_data(subsys);
|
|
|
|
if (subsys->force_stop_gpio > 0 &&
|
|
!subsys_get_crash_status(d->subsys)) {
|
|
gpio_set_value(subsys->force_stop_gpio, 1);
|
|
mdelay(CRASH_STOP_ACK_TO_MS);
|
|
}
|
|
}
|
|
|
|
static irqreturn_t subsys_err_fatal_intr_handler (int irq, void *dev_id)
|
|
{
|
|
struct pil_tz_data *d = subsys_to_data(dev_id);
|
|
|
|
pr_err("Fatal error on %s!\n", d->subsys_desc.name);
|
|
if (subsys_get_crash_status(d->subsys)) {
|
|
pr_err("%s: Ignoring error fatal, restart in progress\n",
|
|
d->subsys_desc.name);
|
|
return IRQ_HANDLED;
|
|
}
|
|
subsys_set_crash_status(d->subsys, true);
|
|
log_failure_reason(d);
|
|
subsystem_restart_dev(d->subsys);
|
|
|
|
return IRQ_HANDLED;
|
|
}
|
|
|
|
static irqreturn_t subsys_wdog_bite_irq_handler(int irq, void *dev_id)
|
|
{
|
|
struct pil_tz_data *d = subsys_to_data(dev_id);
|
|
|
|
if (subsys_get_crash_status(d->subsys))
|
|
return IRQ_HANDLED;
|
|
pr_err("Watchdog bite received from %s!\n", d->subsys_desc.name);
|
|
|
|
if (d->subsys_desc.system_debug &&
|
|
!gpio_get_value(d->subsys_desc.err_fatal_gpio))
|
|
panic("%s: System ramdump requested. Triggering device restart!\n",
|
|
__func__);
|
|
subsys_set_crash_status(d->subsys, true);
|
|
log_failure_reason(d);
|
|
subsystem_restart_dev(d->subsys);
|
|
|
|
return IRQ_HANDLED;
|
|
}
|
|
|
|
static irqreturn_t subsys_stop_ack_intr_handler(int irq, void *dev_id)
|
|
{
|
|
struct pil_tz_data *d = subsys_to_data(dev_id);
|
|
|
|
pr_info("Received stop ack interrupt from %s\n", d->subsys_desc.name);
|
|
complete(&d->stop_ack);
|
|
return IRQ_HANDLED;
|
|
}
|
|
|
|
static int pil_tz_driver_probe(struct platform_device *pdev)
|
|
{
|
|
struct pil_tz_data *d;
|
|
u32 proxy_timeout;
|
|
int len, rc;
|
|
|
|
d = devm_kzalloc(&pdev->dev, sizeof(*d), GFP_KERNEL);
|
|
if (!d)
|
|
return -ENOMEM;
|
|
platform_set_drvdata(pdev, d);
|
|
|
|
if (of_property_read_bool(pdev->dev.of_node, "qcom,pil-no-auth"))
|
|
d->subsys_desc.no_auth = true;
|
|
|
|
d->keep_proxy_regs_on = of_property_read_bool(pdev->dev.of_node,
|
|
"qcom,keep-proxy-regs-on");
|
|
|
|
rc = of_property_read_string(pdev->dev.of_node, "qcom,firmware-name",
|
|
&d->desc.name);
|
|
if (rc)
|
|
return rc;
|
|
|
|
/* Defaulting smem_id to be not present */
|
|
d->smem_id = -1;
|
|
|
|
if (of_find_property(pdev->dev.of_node, "qcom,smem-id", &len)) {
|
|
rc = of_property_read_u32(pdev->dev.of_node, "qcom,smem-id",
|
|
&d->smem_id);
|
|
if (rc) {
|
|
dev_err(&pdev->dev, "Failed to get the smem_id.\n");
|
|
return rc;
|
|
}
|
|
}
|
|
|
|
d->desc.dev = &pdev->dev;
|
|
d->desc.owner = THIS_MODULE;
|
|
d->desc.ops = &pil_ops_trusted;
|
|
|
|
d->desc.proxy_timeout = PROXY_TIMEOUT_MS;
|
|
|
|
rc = of_property_read_u32(pdev->dev.of_node, "qcom,proxy-timeout-ms",
|
|
&proxy_timeout);
|
|
if (!rc)
|
|
d->desc.proxy_timeout = proxy_timeout;
|
|
|
|
if (!d->subsys_desc.no_auth) {
|
|
rc = piltz_resc_init(pdev, d);
|
|
if (rc)
|
|
return -ENOENT;
|
|
|
|
rc = of_property_read_u32(pdev->dev.of_node, "qcom,pas-id",
|
|
&d->pas_id);
|
|
if (rc) {
|
|
dev_err(&pdev->dev, "Failed to find the pas_id.\n");
|
|
return rc;
|
|
}
|
|
scm_pas_init(MSM_BUS_MASTER_CRYPTO_CORE0);
|
|
}
|
|
|
|
rc = pil_desc_init(&d->desc);
|
|
if (rc)
|
|
return rc;
|
|
|
|
init_completion(&d->stop_ack);
|
|
|
|
d->subsys_desc.name = d->desc.name;
|
|
d->subsys_desc.owner = THIS_MODULE;
|
|
d->subsys_desc.dev = &pdev->dev;
|
|
d->subsys_desc.shutdown = subsys_shutdown;
|
|
d->subsys_desc.powerup = subsys_powerup;
|
|
d->subsys_desc.ramdump = subsys_ramdump;
|
|
d->subsys_desc.free_memory = subsys_free_memory;
|
|
d->subsys_desc.crash_shutdown = subsys_crash_shutdown;
|
|
d->subsys_desc.err_fatal_handler = subsys_err_fatal_intr_handler;
|
|
d->subsys_desc.wdog_bite_handler = subsys_wdog_bite_irq_handler;
|
|
d->subsys_desc.stop_ack_handler = subsys_stop_ack_intr_handler;
|
|
|
|
d->ramdump_dev = create_ramdump_device(d->subsys_desc.name,
|
|
&pdev->dev);
|
|
if (!d->ramdump_dev) {
|
|
rc = -ENOMEM;
|
|
goto err_ramdump;
|
|
}
|
|
|
|
d->subsys = subsys_register(&d->subsys_desc);
|
|
if (IS_ERR(d->subsys)) {
|
|
rc = PTR_ERR(d->subsys);
|
|
goto err_subsys;
|
|
}
|
|
|
|
return 0;
|
|
err_subsys:
|
|
destroy_ramdump_device(d->ramdump_dev);
|
|
err_ramdump:
|
|
pil_desc_release(&d->desc);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int pil_tz_driver_exit(struct platform_device *pdev)
|
|
{
|
|
struct pil_tz_data *d = platform_get_drvdata(pdev);
|
|
|
|
subsys_unregister(d->subsys);
|
|
destroy_ramdump_device(d->ramdump_dev);
|
|
pil_desc_release(&d->desc);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct of_device_id pil_tz_match_table[] = {
|
|
{.compatible = "qcom,pil-tz-generic"},
|
|
{}
|
|
};
|
|
|
|
static struct platform_driver pil_tz_driver = {
|
|
.probe = pil_tz_driver_probe,
|
|
.remove = pil_tz_driver_exit,
|
|
.driver = {
|
|
.name = "subsys-pil-tz",
|
|
.of_match_table = pil_tz_match_table,
|
|
.owner = THIS_MODULE,
|
|
},
|
|
};
|
|
|
|
static int __init pil_tz_init(void)
|
|
{
|
|
return platform_driver_register(&pil_tz_driver);
|
|
}
|
|
module_init(pil_tz_init);
|
|
|
|
static void __exit pil_tz_exit(void)
|
|
{
|
|
platform_driver_unregister(&pil_tz_driver);
|
|
}
|
|
module_exit(pil_tz_exit);
|
|
|
|
MODULE_DESCRIPTION("Support for booting subsystems");
|
|
MODULE_LICENSE("GPL v2");
|