|
@@ -32,6 +32,7 @@
|
|
|
#include <linux/soc/qcom/mdt_loader.h>
|
|
|
#include <linux/soc/qcom/smem.h>
|
|
|
#include <linux/soc/qcom/smem_state.h>
|
|
|
+#include <linux/iopoll.h>
|
|
|
|
|
|
#include "remoteproc_internal.h"
|
|
|
#include "qcom_common.h"
|
|
@@ -64,6 +65,8 @@
|
|
|
#define QDSP6SS_RESET_REG 0x014
|
|
|
#define QDSP6SS_GFMUX_CTL_REG 0x020
|
|
|
#define QDSP6SS_PWR_CTL_REG 0x030
|
|
|
+#define QDSP6SS_MEM_PWR_CTL 0x0B0
|
|
|
+#define QDSP6SS_STRAP_ACC 0x110
|
|
|
|
|
|
/* AXI Halt Register Offsets */
|
|
|
#define AXI_HALTREQ_REG 0x0
|
|
@@ -92,6 +95,15 @@
|
|
|
#define QDSS_BHS_ON BIT(21)
|
|
|
#define QDSS_LDO_BYP BIT(22)
|
|
|
|
|
|
+/* QDSP6v56 parameters */
|
|
|
+#define QDSP6v56_LDO_BYP BIT(25)
|
|
|
+#define QDSP6v56_BHS_ON BIT(24)
|
|
|
+#define QDSP6v56_CLAMP_WL BIT(21)
|
|
|
+#define QDSP6v56_CLAMP_QMC_MEM BIT(22)
|
|
|
+#define HALT_CHECK_MAX_LOOPS 200
|
|
|
+#define QDSP6SS_XO_CBCR 0x0038
|
|
|
+#define QDSP6SS_ACC_OVERRIDE_VAL 0x20
|
|
|
+
|
|
|
struct reg_info {
|
|
|
struct regulator *reg;
|
|
|
int uV;
|
|
@@ -110,6 +122,8 @@ struct rproc_hexagon_res {
|
|
|
struct qcom_mss_reg_res *active_supply;
|
|
|
char **proxy_clk_names;
|
|
|
char **active_clk_names;
|
|
|
+ int version;
|
|
|
+ bool need_mem_protection;
|
|
|
};
|
|
|
|
|
|
struct q6v5 {
|
|
@@ -154,6 +168,16 @@ struct q6v5 {
|
|
|
|
|
|
struct qcom_rproc_subdev smd_subdev;
|
|
|
struct qcom_rproc_ssr ssr_subdev;
|
|
|
+ bool need_mem_protection;
|
|
|
+ int mpss_perm;
|
|
|
+ int mba_perm;
|
|
|
+ int version;
|
|
|
+};
|
|
|
+
|
|
|
+enum {
|
|
|
+ MSS_MSM8916,
|
|
|
+ MSS_MSM8974,
|
|
|
+ MSS_MSM8996,
|
|
|
};
|
|
|
|
|
|
static int q6v5_regulator_init(struct device *dev, struct reg_info *regs,
|
|
@@ -289,6 +313,26 @@ static struct resource_table *q6v5_find_rsc_table(struct rproc *rproc,
|
|
|
return &table;
|
|
|
}
|
|
|
|
|
|
+static int q6v5_xfer_mem_ownership(struct q6v5 *qproc, int *current_perm,
|
|
|
+ bool remote_owner, phys_addr_t addr,
|
|
|
+ size_t size)
|
|
|
+{
|
|
|
+ struct qcom_scm_vmperm next;
|
|
|
+
|
|
|
+ if (!qproc->need_mem_protection)
|
|
|
+ return 0;
|
|
|
+ if (remote_owner && *current_perm == BIT(QCOM_SCM_VMID_MSS_MSA))
|
|
|
+ return 0;
|
|
|
+ if (!remote_owner && *current_perm == BIT(QCOM_SCM_VMID_HLOS))
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ next.vmid = remote_owner ? QCOM_SCM_VMID_MSS_MSA : QCOM_SCM_VMID_HLOS;
|
|
|
+ next.perm = remote_owner ? QCOM_SCM_PERM_RW : QCOM_SCM_PERM_RWX;
|
|
|
+
|
|
|
+ return qcom_scm_assign_mem(addr, ALIGN(size, SZ_4K),
|
|
|
+ current_perm, &next, 1);
|
|
|
+}
|
|
|
+
|
|
|
static int q6v5_load(struct rproc *rproc, const struct firmware *fw)
|
|
|
{
|
|
|
struct q6v5 *qproc = rproc->priv;
|
|
@@ -353,33 +397,98 @@ static int q6v5proc_reset(struct q6v5 *qproc)
|
|
|
{
|
|
|
u32 val;
|
|
|
int ret;
|
|
|
+ int i;
|
|
|
|
|
|
- /* Assert resets, stop core */
|
|
|
- val = readl(qproc->reg_base + QDSP6SS_RESET_REG);
|
|
|
- val |= (Q6SS_CORE_ARES | Q6SS_BUS_ARES_ENABLE | Q6SS_STOP_CORE);
|
|
|
- writel(val, qproc->reg_base + QDSP6SS_RESET_REG);
|
|
|
|
|
|
- /* Enable power block headswitch, and wait for it to stabilize */
|
|
|
- val = readl(qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
- val |= QDSS_BHS_ON | QDSS_LDO_BYP;
|
|
|
- writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
- udelay(1);
|
|
|
-
|
|
|
- /*
|
|
|
- * Turn on memories. L2 banks should be done individually
|
|
|
- * to minimize inrush current.
|
|
|
- */
|
|
|
- val = readl(qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
- val |= Q6SS_SLP_RET_N | Q6SS_L2TAG_SLP_NRET_N |
|
|
|
- Q6SS_ETB_SLP_NRET_N | Q6SS_L2DATA_STBY_N;
|
|
|
- writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
- val |= Q6SS_L2DATA_SLP_NRET_N_2;
|
|
|
- writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
- val |= Q6SS_L2DATA_SLP_NRET_N_1;
|
|
|
- writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
- val |= Q6SS_L2DATA_SLP_NRET_N_0;
|
|
|
- writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ if (qproc->version == MSS_MSM8996) {
|
|
|
+ /* Override the ACC value if required */
|
|
|
+ writel(QDSP6SS_ACC_OVERRIDE_VAL,
|
|
|
+ qproc->reg_base + QDSP6SS_STRAP_ACC);
|
|
|
|
|
|
+ /* Assert resets, stop core */
|
|
|
+ val = readl(qproc->reg_base + QDSP6SS_RESET_REG);
|
|
|
+ val |= Q6SS_CORE_ARES | Q6SS_BUS_ARES_ENABLE | Q6SS_STOP_CORE;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_RESET_REG);
|
|
|
+
|
|
|
+ /* BHS require xo cbcr to be enabled */
|
|
|
+ val = readl(qproc->reg_base + QDSP6SS_XO_CBCR);
|
|
|
+ val |= 0x1;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_XO_CBCR);
|
|
|
+
|
|
|
+ /* Read CLKOFF bit to go low indicating CLK is enabled */
|
|
|
+ ret = readl_poll_timeout(qproc->reg_base + QDSP6SS_XO_CBCR,
|
|
|
+ val, !(val & BIT(31)), 1,
|
|
|
+ HALT_CHECK_MAX_LOOPS);
|
|
|
+ if (ret) {
|
|
|
+ dev_err(qproc->dev,
|
|
|
+ "xo cbcr enabling timed out (rc:%d)\n", ret);
|
|
|
+ return ret;
|
|
|
+ }
|
|
|
+ /* Enable power block headswitch and wait for it to stabilize */
|
|
|
+ val = readl(qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ val |= QDSP6v56_BHS_ON;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ val |= readl(qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ udelay(1);
|
|
|
+
|
|
|
+ /* Put LDO in bypass mode */
|
|
|
+ val |= QDSP6v56_LDO_BYP;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+
|
|
|
+ /* Deassert QDSP6 compiler memory clamp */
|
|
|
+ val = readl(qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ val &= ~QDSP6v56_CLAMP_QMC_MEM;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+
|
|
|
+ /* Deassert memory peripheral sleep and L2 memory standby */
|
|
|
+ val |= Q6SS_L2DATA_STBY_N | Q6SS_SLP_RET_N;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+
|
|
|
+ /* Turn on L1, L2, ETB and JU memories 1 at a time */
|
|
|
+ val = readl(qproc->reg_base + QDSP6SS_MEM_PWR_CTL);
|
|
|
+ for (i = 19; i >= 0; i--) {
|
|
|
+ val |= BIT(i);
|
|
|
+ writel(val, qproc->reg_base +
|
|
|
+ QDSP6SS_MEM_PWR_CTL);
|
|
|
+ /*
|
|
|
+ * Read back value to ensure the write is done then
|
|
|
+ * wait for 1us for both memory peripheral and data
|
|
|
+ * array to turn on.
|
|
|
+ */
|
|
|
+ val |= readl(qproc->reg_base + QDSP6SS_MEM_PWR_CTL);
|
|
|
+ udelay(1);
|
|
|
+ }
|
|
|
+ /* Remove word line clamp */
|
|
|
+ val = readl(qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ val &= ~QDSP6v56_CLAMP_WL;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ } else {
|
|
|
+ /* Assert resets, stop core */
|
|
|
+ val = readl(qproc->reg_base + QDSP6SS_RESET_REG);
|
|
|
+ val |= Q6SS_CORE_ARES | Q6SS_BUS_ARES_ENABLE | Q6SS_STOP_CORE;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_RESET_REG);
|
|
|
+
|
|
|
+ /* Enable power block headswitch and wait for it to stabilize */
|
|
|
+ val = readl(qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ val |= QDSS_BHS_ON | QDSS_LDO_BYP;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ val |= readl(qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ udelay(1);
|
|
|
+ /*
|
|
|
+ * Turn on memories. L2 banks should be done individually
|
|
|
+ * to minimize inrush current.
|
|
|
+ */
|
|
|
+ val = readl(qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ val |= Q6SS_SLP_RET_N | Q6SS_L2TAG_SLP_NRET_N |
|
|
|
+ Q6SS_ETB_SLP_NRET_N | Q6SS_L2DATA_STBY_N;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ val |= Q6SS_L2DATA_SLP_NRET_N_2;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ val |= Q6SS_L2DATA_SLP_NRET_N_1;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ val |= Q6SS_L2DATA_SLP_NRET_N_0;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ }
|
|
|
/* Remove IO clamp */
|
|
|
val &= ~Q6SS_CLAMP_IO;
|
|
|
writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
@@ -451,6 +560,8 @@ static int q6v5_mpss_init_image(struct q6v5 *qproc, const struct firmware *fw)
|
|
|
{
|
|
|
unsigned long dma_attrs = DMA_ATTR_FORCE_CONTIGUOUS;
|
|
|
dma_addr_t phys;
|
|
|
+ int mdata_perm;
|
|
|
+ int xferop_ret;
|
|
|
void *ptr;
|
|
|
int ret;
|
|
|
|
|
@@ -462,6 +573,17 @@ static int q6v5_mpss_init_image(struct q6v5 *qproc, const struct firmware *fw)
|
|
|
|
|
|
memcpy(ptr, fw->data, fw->size);
|
|
|
|
|
|
+ /* Hypervisor mapping to access metadata by modem */
|
|
|
+ mdata_perm = BIT(QCOM_SCM_VMID_HLOS);
|
|
|
+ ret = q6v5_xfer_mem_ownership(qproc, &mdata_perm,
|
|
|
+ true, phys, fw->size);
|
|
|
+ if (ret) {
|
|
|
+ dev_err(qproc->dev,
|
|
|
+ "assigning Q6 access to metadata failed: %d\n", ret);
|
|
|
+ ret = -EAGAIN;
|
|
|
+ goto free_dma_attrs;
|
|
|
+ }
|
|
|
+
|
|
|
writel(phys, qproc->rmb_base + RMB_PMI_META_DATA_REG);
|
|
|
writel(RMB_CMD_META_DATA_READY, qproc->rmb_base + RMB_MBA_COMMAND_REG);
|
|
|
|
|
@@ -471,6 +593,14 @@ static int q6v5_mpss_init_image(struct q6v5 *qproc, const struct firmware *fw)
|
|
|
else if (ret < 0)
|
|
|
dev_err(qproc->dev, "MPSS header authentication failed: %d\n", ret);
|
|
|
|
|
|
+ /* Metadata authentication done, remove modem access */
|
|
|
+ xferop_ret = q6v5_xfer_mem_ownership(qproc, &mdata_perm,
|
|
|
+ false, phys, fw->size);
|
|
|
+ if (xferop_ret)
|
|
|
+ dev_warn(qproc->dev,
|
|
|
+ "mdt buffer not reclaimed system may become unstable\n");
|
|
|
+
|
|
|
+free_dma_attrs:
|
|
|
dma_free_attrs(qproc->dev, fw->size, ptr, phys, dma_attrs);
|
|
|
|
|
|
return ret < 0 ? ret : 0;
|
|
@@ -504,7 +634,7 @@ static int q6v5_mpss_load(struct q6v5 *qproc)
|
|
|
bool relocate = false;
|
|
|
char seg_name[10];
|
|
|
ssize_t offset;
|
|
|
- size_t size;
|
|
|
+ size_t size = 0;
|
|
|
void *ptr;
|
|
|
int ret;
|
|
|
int i;
|
|
@@ -542,7 +672,7 @@ static int q6v5_mpss_load(struct q6v5 *qproc)
|
|
|
}
|
|
|
|
|
|
mpss_reloc = relocate ? min_addr : qproc->mpss_phys;
|
|
|
-
|
|
|
+ /* Load firmware segments */
|
|
|
for (i = 0; i < ehdr->e_phnum; i++) {
|
|
|
phdr = &phdrs[i];
|
|
|
|
|
@@ -575,18 +705,24 @@ static int q6v5_mpss_load(struct q6v5 *qproc)
|
|
|
memset(ptr + phdr->p_filesz, 0,
|
|
|
phdr->p_memsz - phdr->p_filesz);
|
|
|
}
|
|
|
-
|
|
|
- size = readl(qproc->rmb_base + RMB_PMI_CODE_LENGTH_REG);
|
|
|
- if (!size) {
|
|
|
- boot_addr = relocate ? qproc->mpss_phys : min_addr;
|
|
|
- writel(boot_addr, qproc->rmb_base + RMB_PMI_CODE_START_REG);
|
|
|
- writel(RMB_CMD_LOAD_READY, qproc->rmb_base + RMB_MBA_COMMAND_REG);
|
|
|
- }
|
|
|
-
|
|
|
size += phdr->p_memsz;
|
|
|
- writel(size, qproc->rmb_base + RMB_PMI_CODE_LENGTH_REG);
|
|
|
}
|
|
|
|
|
|
+ /* Transfer ownership of modem ddr region to q6 */
|
|
|
+ ret = q6v5_xfer_mem_ownership(qproc, &qproc->mpss_perm, true,
|
|
|
+ qproc->mpss_phys, qproc->mpss_size);
|
|
|
+ if (ret) {
|
|
|
+ dev_err(qproc->dev,
|
|
|
+ "assigning Q6 access to mpss memory failed: %d\n", ret);
|
|
|
+ ret = -EAGAIN;
|
|
|
+ goto release_firmware;
|
|
|
+ }
|
|
|
+
|
|
|
+ boot_addr = relocate ? qproc->mpss_phys : min_addr;
|
|
|
+ writel(boot_addr, qproc->rmb_base + RMB_PMI_CODE_START_REG);
|
|
|
+ writel(RMB_CMD_LOAD_READY, qproc->rmb_base + RMB_MBA_COMMAND_REG);
|
|
|
+ writel(size, qproc->rmb_base + RMB_PMI_CODE_LENGTH_REG);
|
|
|
+
|
|
|
ret = q6v5_rmb_mba_wait(qproc, RMB_MBA_AUTH_COMPLETE, 10000);
|
|
|
if (ret == -ETIMEDOUT)
|
|
|
dev_err(qproc->dev, "MPSS authentication timed out\n");
|
|
@@ -602,6 +738,7 @@ release_firmware:
|
|
|
static int q6v5_start(struct rproc *rproc)
|
|
|
{
|
|
|
struct q6v5 *qproc = (struct q6v5 *)rproc->priv;
|
|
|
+ int xfermemop_ret;
|
|
|
int ret;
|
|
|
|
|
|
ret = q6v5_regulator_enable(qproc, qproc->proxy_regs,
|
|
@@ -637,11 +774,22 @@ static int q6v5_start(struct rproc *rproc)
|
|
|
goto assert_reset;
|
|
|
}
|
|
|
|
|
|
+ /* Assign MBA image access in DDR to q6 */
|
|
|
+ xfermemop_ret = q6v5_xfer_mem_ownership(qproc, &qproc->mba_perm, true,
|
|
|
+ qproc->mba_phys,
|
|
|
+ qproc->mba_size);
|
|
|
+ if (xfermemop_ret) {
|
|
|
+ dev_err(qproc->dev,
|
|
|
+ "assigning Q6 access to mba memory failed: %d\n",
|
|
|
+ xfermemop_ret);
|
|
|
+ goto disable_active_clks;
|
|
|
+ }
|
|
|
+
|
|
|
writel(qproc->mba_phys, qproc->rmb_base + RMB_MBA_IMAGE_REG);
|
|
|
|
|
|
ret = q6v5proc_reset(qproc);
|
|
|
if (ret)
|
|
|
- goto halt_axi_ports;
|
|
|
+ goto reclaim_mba;
|
|
|
|
|
|
ret = q6v5_rmb_mba_wait(qproc, 0, 5000);
|
|
|
if (ret == -ETIMEDOUT) {
|
|
@@ -658,16 +806,22 @@ static int q6v5_start(struct rproc *rproc)
|
|
|
|
|
|
ret = q6v5_mpss_load(qproc);
|
|
|
if (ret)
|
|
|
- goto halt_axi_ports;
|
|
|
+ goto reclaim_mpss;
|
|
|
|
|
|
ret = wait_for_completion_timeout(&qproc->start_done,
|
|
|
msecs_to_jiffies(5000));
|
|
|
if (ret == 0) {
|
|
|
dev_err(qproc->dev, "start timed out\n");
|
|
|
ret = -ETIMEDOUT;
|
|
|
- goto halt_axi_ports;
|
|
|
+ goto reclaim_mpss;
|
|
|
}
|
|
|
|
|
|
+ xfermemop_ret = q6v5_xfer_mem_ownership(qproc, &qproc->mba_perm, false,
|
|
|
+ qproc->mba_phys,
|
|
|
+ qproc->mba_size);
|
|
|
+ if (xfermemop_ret)
|
|
|
+ dev_err(qproc->dev,
|
|
|
+ "Failed to reclaim mba buffer system may become unstable\n");
|
|
|
qproc->running = true;
|
|
|
|
|
|
q6v5_clk_disable(qproc->dev, qproc->proxy_clks,
|
|
@@ -677,12 +831,30 @@ static int q6v5_start(struct rproc *rproc)
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
+reclaim_mpss:
|
|
|
+ xfermemop_ret = q6v5_xfer_mem_ownership(qproc, &qproc->mpss_perm,
|
|
|
+ false, qproc->mpss_phys,
|
|
|
+ qproc->mpss_size);
|
|
|
+ WARN_ON(xfermemop_ret);
|
|
|
+
|
|
|
halt_axi_ports:
|
|
|
q6v5proc_halt_axi_port(qproc, qproc->halt_map, qproc->halt_q6);
|
|
|
q6v5proc_halt_axi_port(qproc, qproc->halt_map, qproc->halt_modem);
|
|
|
q6v5proc_halt_axi_port(qproc, qproc->halt_map, qproc->halt_nc);
|
|
|
+
|
|
|
+reclaim_mba:
|
|
|
+ xfermemop_ret = q6v5_xfer_mem_ownership(qproc, &qproc->mba_perm, false,
|
|
|
+ qproc->mba_phys,
|
|
|
+ qproc->mba_size);
|
|
|
+ if (xfermemop_ret) {
|
|
|
+ dev_err(qproc->dev,
|
|
|
+ "Failed to reclaim mba buffer, system may become unstable\n");
|
|
|
+ }
|
|
|
+
|
|
|
+disable_active_clks:
|
|
|
q6v5_clk_disable(qproc->dev, qproc->active_clks,
|
|
|
qproc->active_clk_count);
|
|
|
+
|
|
|
assert_reset:
|
|
|
reset_control_assert(qproc->mss_restart);
|
|
|
disable_vdd:
|
|
@@ -702,6 +874,7 @@ static int q6v5_stop(struct rproc *rproc)
|
|
|
{
|
|
|
struct q6v5 *qproc = (struct q6v5 *)rproc->priv;
|
|
|
int ret;
|
|
|
+ u32 val;
|
|
|
|
|
|
qproc->running = false;
|
|
|
|
|
@@ -718,6 +891,20 @@ static int q6v5_stop(struct rproc *rproc)
|
|
|
q6v5proc_halt_axi_port(qproc, qproc->halt_map, qproc->halt_q6);
|
|
|
q6v5proc_halt_axi_port(qproc, qproc->halt_map, qproc->halt_modem);
|
|
|
q6v5proc_halt_axi_port(qproc, qproc->halt_map, qproc->halt_nc);
|
|
|
+ if (qproc->version == MSS_MSM8996) {
|
|
|
+ /*
|
|
|
+ * To avoid high MX current during LPASS/MSS restart.
|
|
|
+ */
|
|
|
+ val = readl(qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ val |= Q6SS_CLAMP_IO | QDSP6v56_CLAMP_WL |
|
|
|
+ QDSP6v56_CLAMP_QMC_MEM;
|
|
|
+ writel(val, qproc->reg_base + QDSP6SS_PWR_CTL_REG);
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ ret = q6v5_xfer_mem_ownership(qproc, &qproc->mpss_perm, false,
|
|
|
+ qproc->mpss_phys, qproc->mpss_size);
|
|
|
+ WARN_ON(ret);
|
|
|
|
|
|
reset_control_assert(qproc->mss_restart);
|
|
|
q6v5_clk_disable(qproc->dev, qproc->active_clks,
|
|
@@ -1017,6 +1204,8 @@ static int q6v5_probe(struct platform_device *pdev)
|
|
|
if (ret)
|
|
|
goto free_rproc;
|
|
|
|
|
|
+ qproc->version = desc->version;
|
|
|
+ qproc->need_mem_protection = desc->need_mem_protection;
|
|
|
ret = q6v5_request_irq(qproc, pdev, "wdog", q6v5_wdog_interrupt);
|
|
|
if (ret < 0)
|
|
|
goto free_rproc;
|
|
@@ -1038,7 +1227,8 @@ static int q6v5_probe(struct platform_device *pdev)
|
|
|
ret = PTR_ERR(qproc->state);
|
|
|
goto free_rproc;
|
|
|
}
|
|
|
-
|
|
|
+ qproc->mpss_perm = BIT(QCOM_SCM_VMID_HLOS);
|
|
|
+ qproc->mba_perm = BIT(QCOM_SCM_VMID_HLOS);
|
|
|
qcom_add_smd_subdev(rproc, &qproc->smd_subdev);
|
|
|
qcom_add_ssr_subdev(rproc, &qproc->ssr_subdev, "mpss");
|
|
|
|
|
@@ -1067,6 +1257,24 @@ static int q6v5_remove(struct platform_device *pdev)
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
+static const struct rproc_hexagon_res msm8996_mss = {
|
|
|
+ .hexagon_mba_image = "mba.mbn",
|
|
|
+ .proxy_clk_names = (char*[]){
|
|
|
+ "xo",
|
|
|
+ "pnoc",
|
|
|
+ NULL
|
|
|
+ },
|
|
|
+ .active_clk_names = (char*[]){
|
|
|
+ "iface",
|
|
|
+ "bus",
|
|
|
+ "mem",
|
|
|
+ "gpll0_mss_clk",
|
|
|
+ NULL
|
|
|
+ },
|
|
|
+ .need_mem_protection = true,
|
|
|
+ .version = MSS_MSM8996,
|
|
|
+};
|
|
|
+
|
|
|
static const struct rproc_hexagon_res msm8916_mss = {
|
|
|
.hexagon_mba_image = "mba.mbn",
|
|
|
.proxy_supply = (struct qcom_mss_reg_res[]) {
|
|
@@ -1094,6 +1302,8 @@ static const struct rproc_hexagon_res msm8916_mss = {
|
|
|
"mem",
|
|
|
NULL
|
|
|
},
|
|
|
+ .need_mem_protection = false,
|
|
|
+ .version = MSS_MSM8916,
|
|
|
};
|
|
|
|
|
|
static const struct rproc_hexagon_res msm8974_mss = {
|
|
@@ -1131,12 +1341,15 @@ static const struct rproc_hexagon_res msm8974_mss = {
|
|
|
"mem",
|
|
|
NULL
|
|
|
},
|
|
|
+ .need_mem_protection = false,
|
|
|
+ .version = MSS_MSM8974,
|
|
|
};
|
|
|
|
|
|
static const struct of_device_id q6v5_of_match[] = {
|
|
|
{ .compatible = "qcom,q6v5-pil", .data = &msm8916_mss},
|
|
|
{ .compatible = "qcom,msm8916-mss-pil", .data = &msm8916_mss},
|
|
|
{ .compatible = "qcom,msm8974-mss-pil", .data = &msm8974_mss},
|
|
|
+ { .compatible = "qcom,msm8996-mss-pil", .data = &msm8996_mss},
|
|
|
{ },
|
|
|
};
|
|
|
MODULE_DEVICE_TABLE(of, q6v5_of_match);
|