mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-08 14:23:19 +00:00
remoteproc updates for v5.19
This fixes a race condition in the user space interface for starting and stopping remote processors, it makes the ELF loader properly skip zero memsz segments and it cleans up the debugfs tracefile code a bit by not checking for errors. It introduces support for controlling the audio DSP on Qualcomm MSM8226, as well as audio and compute DSPs on Qualcomm SC8280XP. It makes it possible to specify the firmware path for Mediatek's remote processors, fixes a double free in the SCP driver and addresses an issue with the SRAM initialization on MT8195. Lastly it deprecates the custom ELF loader in the iMX remoteproc driver, in favor of using the shared one. -----BEGIN PGP SIGNATURE----- iQJPBAABCAA5FiEEBd4DzF816k8JZtUlCx85Pw2ZrcUFAmKXWMAbHGJqb3JuLmFu ZGVyc3NvbkBsaW5hcm8ub3JnAAoJEAsfOT8Nma3F/lsQAKhQZs6uMlmv/SUt0BjM wWdbH/KgfeBdLeD+OZCigwVvBKk5HNSQd+K+KIpAHUyScKv5JKrXRYbmzh3eylpk Wq8ua5rRGwBBpvhlkUC6l4VvaoFHynNLwsSL+zbcBMXZvMZ793WDPsK9bAsRKWdH Zy0q1/B+T/BjaM0wfDEZljWI7JxYbAVnr/BDQztNTydJqRPysuBi9KGP7yD62rpt x+RtnwOMMGkquwrlscQpHFuj4T92UD5/BHmaT16QkDpEkPnA8LnUye97xNwPfSas TQi9UyhhMAX9bzZe4LTIbEjO6UJrmw6JPlhsz8HG7xhfE+kQ+navD4NH/eLJziEM 4UFBMlyaSBKTHcpbmmCW+UOzYh3SRZ+nrZupAZAddThAStkdE5FB4uY3Yd7B6aSQ H+gc/RfRHsOiEqEO3NQ3vMbVODa2bijbsJ42p9jfphgfejIjkR1jzvCVaXzQNX2x tlzxF/3cvmFoH2G/wA9JXkexbNCA3WSWXv7SSXhNufLGiME5lS/DCDf3MM6f7OiQ L5sypoJiKbEwzYfD67IlnQ5htdwHdhiSFr6Np86FwgH8Ninaqgk8PMLll/Uefjpj BQ+9iPRFuWEjB//miBgvfzZ7jIO/OzfaarMkI05xRF9V4ZMWLDBjSGR7eBJukC+e GZ0lkc6/q89riD5NZvGWBXrc =UwTC -----END PGP SIGNATURE----- Merge tag 'rproc-v5.19' of git://git.kernel.org/pub/scm/linux/kernel/git/remoteproc/linux Pull remoteproc updates from Bjorn Andersson: "This fixes a race condition in the user space interface for starting and stopping remote processors, it makes the ELF loader properly skip zero memsz segments and it cleans up the debugfs tracefile code a bit by not checking for errors. It introduces support for controlling the audio DSP on Qualcomm MSM8226, as well as audio and compute DSPs on Qualcomm SC8280XP. It makes it possible to specify the firmware path for Mediatek's remote processors, fixes a double free in the SCP driver and addresses an issue with the SRAM initialization on MT8195. Lastly it deprecates the custom ELF loader in the iMX remoteproc driver, in favor of using the shared one" * tag 'rproc-v5.19' of git://git.kernel.org/pub/scm/linux/kernel/git/remoteproc/linux: (21 commits) dt-bindings: remoteproc: mediatek: Add optional memory-region to mtk,scp dt-bindings: remoteproc: mediatek: Make l1tcm reg exclusive to mt819x dt-bindings: remoteproc: st,stm32-rproc: Fix phandle-array parameters description remoteproc: imx_rproc: Support i.MX93 dt-bindings: remoteproc: imx_rproc: Support i.MX93 remoteproc: qcom: pas: Add MSM8226 ADSP support dt-bindings: remoteproc: qcom: pas: Add MSM8226 adsp remoteproc: mediatek: Allow reading firmware-name from DT dt-bindings: remoteproc: mediatek: Add firmware-name property remoteproc: qcom: pas: Add sc8280xp remoteprocs dt-bindings: remoteproc: qcom: pas: Add sc8280xp adsp and nsp pair dt-bindings: remoteproc: mediatek: Add interrupts property to mtk,scp remoteproc: imx_rproc: Ignore create mem entry for resource table remoteproc: core: Move state checking to remoteproc_core remoteproc: core: Remove state checking before calling rproc_boot() remoteproc: imx_dsp_rproc: Make rsc_table optional remoteproc: imx_dsp_rproc: use common rproc_elf_load_segments remoteproc: elf_loader: skip segment with memsz as zero remoteproc: mtk_scp: Fix a potential double free remoteproc: Don't bother checking the return value of debugfs_create* ...
This commit is contained in:
commit
f634b63d43
@ -15,14 +15,15 @@ maintainers:
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- fsl,imx8mq-cm4
|
||||
- fsl,imx6sx-cm4
|
||||
- fsl,imx7d-cm4
|
||||
- fsl,imx7ulp-cm4
|
||||
- fsl,imx8mm-cm4
|
||||
- fsl,imx8mn-cm7
|
||||
- fsl,imx8mp-cm7
|
||||
- fsl,imx8mq-cm4
|
||||
- fsl,imx8ulp-cm33
|
||||
- fsl,imx7d-cm4
|
||||
- fsl,imx7ulp-cm4
|
||||
- fsl,imx6sx-cm4
|
||||
- fsl,imx93-cm33
|
||||
|
||||
clocks:
|
||||
maxItems: 1
|
||||
|
@ -23,11 +23,13 @@ properties:
|
||||
|
||||
reg:
|
||||
description:
|
||||
Should contain the address ranges for memory regions SRAM, CFG, and
|
||||
L1TCM.
|
||||
Should contain the address ranges for memory regions SRAM, CFG, and,
|
||||
on some platforms, L1TCM.
|
||||
minItems: 2
|
||||
maxItems: 3
|
||||
|
||||
reg-names:
|
||||
minItems: 2
|
||||
items:
|
||||
- const: sram
|
||||
- const: cfg
|
||||
@ -42,21 +44,48 @@ properties:
|
||||
clock-names:
|
||||
const: main
|
||||
|
||||
interrupts:
|
||||
maxItems: 1
|
||||
|
||||
firmware-name:
|
||||
$ref: /schemas/types.yaml#/definitions/string
|
||||
description:
|
||||
If present, name (or relative path) of the file within the
|
||||
firmware search path containing the firmware image used when
|
||||
initializing SCP.
|
||||
|
||||
memory-region:
|
||||
maxItems: 1
|
||||
|
||||
required:
|
||||
- compatible
|
||||
- reg
|
||||
- reg-names
|
||||
|
||||
if:
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- mediatek,mt8183-scp
|
||||
- mediatek,mt8192-scp
|
||||
then:
|
||||
required:
|
||||
- clocks
|
||||
- clock-names
|
||||
allOf:
|
||||
- if:
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- mediatek,mt8183-scp
|
||||
- mediatek,mt8192-scp
|
||||
then:
|
||||
required:
|
||||
- clocks
|
||||
- clock-names
|
||||
|
||||
- if:
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- mediatek,mt8183-scp
|
||||
- mediatek,mt8186-scp
|
||||
then:
|
||||
properties:
|
||||
reg:
|
||||
maxItems: 2
|
||||
reg-names:
|
||||
maxItems: 2
|
||||
|
||||
additionalProperties:
|
||||
type: object
|
||||
@ -76,10 +105,10 @@ additionalProperties:
|
||||
|
||||
examples:
|
||||
- |
|
||||
#include <dt-bindings/clock/mt8183-clk.h>
|
||||
#include <dt-bindings/clock/mt8192-clk.h>
|
||||
|
||||
scp@10500000 {
|
||||
compatible = "mediatek,mt8183-scp";
|
||||
compatible = "mediatek,mt8192-scp";
|
||||
reg = <0x10500000 0x80000>,
|
||||
<0x10700000 0x8000>,
|
||||
<0x10720000 0xe0000>;
|
||||
|
@ -16,6 +16,7 @@ description:
|
||||
properties:
|
||||
compatible:
|
||||
enum:
|
||||
- qcom,msm8226-adsp-pil
|
||||
- qcom,msm8974-adsp-pil
|
||||
- qcom,msm8996-adsp-pil
|
||||
- qcom,msm8996-slpi-pil
|
||||
@ -29,6 +30,9 @@ properties:
|
||||
- qcom,sc8180x-adsp-pas
|
||||
- qcom,sc8180x-cdsp-pas
|
||||
- qcom,sc8180x-mpss-pas
|
||||
- qcom,sc8280xp-adsp-pas
|
||||
- qcom,sc8280xp-nsp0-pas
|
||||
- qcom,sc8280xp-nsp1-pas
|
||||
- qcom,sdm660-adsp-pas
|
||||
- qcom,sdm845-adsp-pas
|
||||
- qcom,sdm845-cdsp-pas
|
||||
@ -159,6 +163,7 @@ allOf:
|
||||
compatible:
|
||||
contains:
|
||||
enum:
|
||||
- qcom,msm8226-adsp-pil
|
||||
- qcom,msm8974-adsp-pil
|
||||
- qcom,msm8996-adsp-pil
|
||||
- qcom,msm8996-slpi-pil
|
||||
@ -169,6 +174,9 @@ allOf:
|
||||
- qcom,sc8180x-adsp-pas
|
||||
- qcom,sc8180x-cdsp-pas
|
||||
- qcom,sc8180x-mpss-pas
|
||||
- qcom,sc8280xp-adsp-pas
|
||||
- qcom,sc8280xp-nsp0-pas
|
||||
- qcom,sc8280xp-nsp1-pas
|
||||
- qcom,sdm845-adsp-pas
|
||||
- qcom,sdm845-cdsp-pas
|
||||
- qcom,sm6350-adsp-pas
|
||||
@ -274,6 +282,7 @@ allOf:
|
||||
compatible:
|
||||
contains:
|
||||
enum:
|
||||
- qcom,msm8226-adsp-pil
|
||||
- qcom,msm8974-adsp-pil
|
||||
- qcom,msm8996-adsp-pil
|
||||
- qcom,msm8996-slpi-pil
|
||||
@ -284,6 +293,9 @@ allOf:
|
||||
- qcom,qcs404-wcss-pas
|
||||
- qcom,sc8180x-adsp-pas
|
||||
- qcom,sc8180x-cdsp-pas
|
||||
- qcom,sc8280xp-adsp-pas
|
||||
- qcom,sc8280xp-nsp0-pas
|
||||
- qcom,sc8280xp-nsp1-pas
|
||||
- qcom,sdm845-adsp-pas
|
||||
- qcom,sdm845-cdsp-pas
|
||||
- qcom,sm6350-adsp-pas
|
||||
@ -364,6 +376,7 @@ allOf:
|
||||
compatible:
|
||||
contains:
|
||||
enum:
|
||||
- qcom,msm8226-adsp-pil
|
||||
- qcom,msm8996-adsp-pil
|
||||
- qcom,msm8998-adsp-pas
|
||||
then:
|
||||
@ -471,6 +484,7 @@ allOf:
|
||||
enum:
|
||||
- qcom,sc8180x-adsp-pas
|
||||
- qcom,sc8180x-cdsp-pas
|
||||
- qcom,sc8280xp-adsp-pas
|
||||
- qcom,sm6350-adsp-pas
|
||||
- qcom,sm8150-slpi-pas
|
||||
- qcom,sm8250-adsp-pas
|
||||
@ -508,6 +522,22 @@ allOf:
|
||||
- const: cx
|
||||
- const: mxc
|
||||
|
||||
- if:
|
||||
properties:
|
||||
compatible:
|
||||
contains:
|
||||
enum:
|
||||
- qcom,sc8280xp-nsp0-pas
|
||||
- qcom,sc8280xp-nsp1-pas
|
||||
then:
|
||||
properties:
|
||||
power-domains:
|
||||
items:
|
||||
- description: NSP power domain
|
||||
power-domain-names:
|
||||
items:
|
||||
- const: nsp
|
||||
|
||||
- if:
|
||||
properties:
|
||||
compatible:
|
||||
@ -546,6 +576,7 @@ allOf:
|
||||
compatible:
|
||||
contains:
|
||||
enum:
|
||||
- qcom,msm8226-adsp-pil
|
||||
- qcom,msm8974-adsp-pil
|
||||
- qcom,msm8996-adsp-pil
|
||||
- qcom,msm8996-slpi-pil
|
||||
|
@ -43,8 +43,8 @@ properties:
|
||||
items:
|
||||
- items:
|
||||
- description: Phandle of syscon block
|
||||
- description: FIXME
|
||||
- description: FIXME
|
||||
- description: The offset of the trust zone setting register
|
||||
- description: The field mask of the trust zone state
|
||||
|
||||
interrupts:
|
||||
description: Should contain the WWDG1 watchdog reset interrupt
|
||||
@ -101,8 +101,8 @@ properties:
|
||||
items:
|
||||
- items:
|
||||
- description: Phandle of syscon block
|
||||
- description: FIXME
|
||||
- description: FIXME
|
||||
- description: The offset of the power setting register
|
||||
- description: The field mask of the PDDS selection
|
||||
|
||||
st,syscfg-m4-state:
|
||||
$ref: "/schemas/types.yaml#/definitions/phandle-array"
|
||||
@ -111,8 +111,8 @@ properties:
|
||||
items:
|
||||
- items:
|
||||
- description: Phandle of syscon block with the tamp register
|
||||
- description: FIXME
|
||||
- description: FIXME
|
||||
- description: The offset of the tamp register
|
||||
- description: The field mask of the Cortex-M4 state
|
||||
|
||||
st,syscfg-rsc-tbl:
|
||||
$ref: "/schemas/types.yaml#/definitions/phandle-array"
|
||||
@ -122,8 +122,8 @@ properties:
|
||||
items:
|
||||
- items:
|
||||
- description: Phandle of syscon block with the tamp register
|
||||
- description: FIXME
|
||||
- description: FIXME
|
||||
- description: The offset of the tamp register
|
||||
- description: The field mask of the Cortex-M4 resource table address
|
||||
|
||||
st,auto-boot:
|
||||
$ref: /schemas/types.yaml#/definitions/flag
|
||||
|
@ -649,99 +649,6 @@ static int imx_dsp_rproc_add_carveout(struct imx_dsp_rproc *priv)
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* imx_dsp_rproc_elf_load_segments() - load firmware segments to memory
|
||||
* @rproc: remote processor which will be booted using these fw segments
|
||||
* @fw: the ELF firmware image
|
||||
*
|
||||
* This function specially checks if memsz is zero or not, otherwise it
|
||||
* is mostly same as rproc_elf_load_segments().
|
||||
*/
|
||||
static int imx_dsp_rproc_elf_load_segments(struct rproc *rproc,
|
||||
const struct firmware *fw)
|
||||
{
|
||||
struct device *dev = &rproc->dev;
|
||||
u8 class = fw_elf_get_class(fw);
|
||||
u32 elf_phdr_get_size = elf_size_of_phdr(class);
|
||||
const u8 *elf_data = fw->data;
|
||||
const void *ehdr, *phdr;
|
||||
int i, ret = 0;
|
||||
u16 phnum;
|
||||
|
||||
ehdr = elf_data;
|
||||
phnum = elf_hdr_get_e_phnum(class, ehdr);
|
||||
phdr = elf_data + elf_hdr_get_e_phoff(class, ehdr);
|
||||
|
||||
/* go through the available ELF segments */
|
||||
for (i = 0; i < phnum; i++, phdr += elf_phdr_get_size) {
|
||||
u64 da = elf_phdr_get_p_paddr(class, phdr);
|
||||
u64 memsz = elf_phdr_get_p_memsz(class, phdr);
|
||||
u64 filesz = elf_phdr_get_p_filesz(class, phdr);
|
||||
u64 offset = elf_phdr_get_p_offset(class, phdr);
|
||||
u32 type = elf_phdr_get_p_type(class, phdr);
|
||||
void *ptr;
|
||||
|
||||
/*
|
||||
* There is a case that with PT_LOAD type, the
|
||||
* filesz = memsz = 0. If memsz = 0, rproc_da_to_va
|
||||
* should return NULL ptr, then error is returned.
|
||||
* So this case should be skipped from the loop.
|
||||
* Add !memsz checking here.
|
||||
*/
|
||||
if (type != PT_LOAD || !memsz)
|
||||
continue;
|
||||
|
||||
dev_dbg(dev, "phdr: type %d da 0x%llx memsz 0x%llx filesz 0x%llx\n",
|
||||
type, da, memsz, filesz);
|
||||
|
||||
if (filesz > memsz) {
|
||||
dev_err(dev, "bad phdr filesz 0x%llx memsz 0x%llx\n",
|
||||
filesz, memsz);
|
||||
ret = -EINVAL;
|
||||
break;
|
||||
}
|
||||
|
||||
if (offset + filesz > fw->size) {
|
||||
dev_err(dev, "truncated fw: need 0x%llx avail 0x%zx\n",
|
||||
offset + filesz, fw->size);
|
||||
ret = -EINVAL;
|
||||
break;
|
||||
}
|
||||
|
||||
if (!rproc_u64_fit_in_size_t(memsz)) {
|
||||
dev_err(dev, "size (%llx) does not fit in size_t type\n",
|
||||
memsz);
|
||||
ret = -EOVERFLOW;
|
||||
break;
|
||||
}
|
||||
|
||||
/* grab the kernel address for this device address */
|
||||
ptr = rproc_da_to_va(rproc, da, memsz, NULL);
|
||||
if (!ptr) {
|
||||
dev_err(dev, "bad phdr da 0x%llx mem 0x%llx\n", da,
|
||||
memsz);
|
||||
ret = -EINVAL;
|
||||
break;
|
||||
}
|
||||
|
||||
/* put the segment where the remote processor expects it */
|
||||
if (filesz)
|
||||
memcpy(ptr, elf_data + offset, filesz);
|
||||
|
||||
/*
|
||||
* Zero out remaining memory for this segment.
|
||||
*
|
||||
* This isn't strictly required since dma_alloc_coherent already
|
||||
* did this for us. albeit harmless, we may consider removing
|
||||
* this.
|
||||
*/
|
||||
if (memsz > filesz)
|
||||
memset(ptr + filesz, 0, memsz - filesz);
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* Prepare function for rproc_ops */
|
||||
static int imx_dsp_rproc_prepare(struct rproc *rproc)
|
||||
{
|
||||
@ -802,14 +709,22 @@ static void imx_dsp_rproc_kick(struct rproc *rproc, int vqid)
|
||||
dev_err(dev, "%s: failed (%d, err:%d)\n", __func__, vqid, err);
|
||||
}
|
||||
|
||||
static int imx_dsp_rproc_parse_fw(struct rproc *rproc, const struct firmware *fw)
|
||||
{
|
||||
if (rproc_elf_load_rsc_table(rproc, fw))
|
||||
dev_warn(&rproc->dev, "no resource table found for this firmware\n");
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const struct rproc_ops imx_dsp_rproc_ops = {
|
||||
.prepare = imx_dsp_rproc_prepare,
|
||||
.unprepare = imx_dsp_rproc_unprepare,
|
||||
.start = imx_dsp_rproc_start,
|
||||
.stop = imx_dsp_rproc_stop,
|
||||
.kick = imx_dsp_rproc_kick,
|
||||
.load = imx_dsp_rproc_elf_load_segments,
|
||||
.parse_fw = rproc_elf_load_rsc_table,
|
||||
.load = rproc_elf_load_segments,
|
||||
.parse_fw = imx_dsp_rproc_parse_fw,
|
||||
.sanity_check = rproc_elf_sanity_check,
|
||||
.get_boot_addr = rproc_elf_get_boot_addr,
|
||||
};
|
||||
|
@ -91,6 +91,32 @@ struct imx_rproc {
|
||||
void __iomem *rsc_table;
|
||||
};
|
||||
|
||||
static const struct imx_rproc_att imx_rproc_att_imx93[] = {
|
||||
/* dev addr , sys addr , size , flags */
|
||||
/* TCM CODE NON-SECURE */
|
||||
{ 0x0FFC0000, 0x201C0000, 0x00020000, ATT_OWN | ATT_IOMEM },
|
||||
{ 0x0FFE0000, 0x201E0000, 0x00020000, ATT_OWN | ATT_IOMEM },
|
||||
|
||||
/* TCM CODE SECURE */
|
||||
{ 0x1FFC0000, 0x201C0000, 0x00020000, ATT_OWN | ATT_IOMEM },
|
||||
{ 0x1FFE0000, 0x201E0000, 0x00020000, ATT_OWN | ATT_IOMEM },
|
||||
|
||||
/* TCM SYS NON-SECURE*/
|
||||
{ 0x20000000, 0x20200000, 0x00020000, ATT_OWN | ATT_IOMEM },
|
||||
{ 0x20020000, 0x20220000, 0x00020000, ATT_OWN | ATT_IOMEM },
|
||||
|
||||
/* TCM SYS SECURE*/
|
||||
{ 0x30000000, 0x20200000, 0x00020000, ATT_OWN | ATT_IOMEM },
|
||||
{ 0x30020000, 0x20220000, 0x00020000, ATT_OWN | ATT_IOMEM },
|
||||
|
||||
/* DDR */
|
||||
{ 0x80000000, 0x80000000, 0x10000000, 0 },
|
||||
{ 0x90000000, 0x80000000, 0x10000000, 0 },
|
||||
|
||||
{ 0xC0000000, 0xa0000000, 0x10000000, 0 },
|
||||
{ 0xD0000000, 0xa0000000, 0x10000000, 0 },
|
||||
};
|
||||
|
||||
static const struct imx_rproc_att imx_rproc_att_imx8mn[] = {
|
||||
/* dev addr , sys addr , size , flags */
|
||||
/* ITCM */
|
||||
@ -261,6 +287,12 @@ static const struct imx_rproc_dcfg imx_rproc_cfg_imx6sx = {
|
||||
.method = IMX_RPROC_MMIO,
|
||||
};
|
||||
|
||||
static const struct imx_rproc_dcfg imx_rproc_cfg_imx93 = {
|
||||
.att = imx_rproc_att_imx93,
|
||||
.att_size = ARRAY_SIZE(imx_rproc_att_imx93),
|
||||
.method = IMX_RPROC_SMC,
|
||||
};
|
||||
|
||||
static int imx_rproc_start(struct rproc *rproc)
|
||||
{
|
||||
struct imx_rproc *priv = rproc->priv;
|
||||
@ -423,6 +455,9 @@ static int imx_rproc_prepare(struct rproc *rproc)
|
||||
if (!strcmp(it.node->name, "vdev0buffer"))
|
||||
continue;
|
||||
|
||||
if (!strcmp(it.node->name, "rsc-table"))
|
||||
continue;
|
||||
|
||||
rmem = of_reserved_mem_lookup(it.node);
|
||||
if (!rmem) {
|
||||
dev_err(priv->dev, "unable to acquire memory-region\n");
|
||||
@ -821,6 +856,7 @@ static const struct of_device_id imx_rproc_of_match[] = {
|
||||
{ .compatible = "fsl,imx8mn-cm7", .data = &imx_rproc_cfg_imx8mn },
|
||||
{ .compatible = "fsl,imx8mp-cm7", .data = &imx_rproc_cfg_imx8mn },
|
||||
{ .compatible = "fsl,imx8ulp-cm33", .data = &imx_rproc_cfg_imx8ulp },
|
||||
{ .compatible = "fsl,imx93-cm33", .data = &imx_rproc_cfg_imx93 },
|
||||
{},
|
||||
};
|
||||
MODULE_DEVICE_TABLE(of, imx_rproc_of_match);
|
||||
|
@ -54,6 +54,8 @@
|
||||
#define MT8192_CORE0_WDT_IRQ 0x10030
|
||||
#define MT8192_CORE0_WDT_CFG 0x10034
|
||||
|
||||
#define MT8195_L1TCM_SRAM_PDN_RESERVED_RSI_BITS GENMASK(7, 4)
|
||||
|
||||
#define SCP_FW_VER_LEN 32
|
||||
#define SCP_SHARE_BUFFER_SIZE 288
|
||||
|
||||
|
@ -365,22 +365,22 @@ static int mt8183_scp_before_load(struct mtk_scp *scp)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void mt8192_power_on_sram(void __iomem *addr)
|
||||
static void scp_sram_power_on(void __iomem *addr, u32 reserved_mask)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 31; i >= 0; i--)
|
||||
writel(GENMASK(i, 0), addr);
|
||||
writel(GENMASK(i, 0) & ~reserved_mask, addr);
|
||||
writel(0, addr);
|
||||
}
|
||||
|
||||
static void mt8192_power_off_sram(void __iomem *addr)
|
||||
static void scp_sram_power_off(void __iomem *addr, u32 reserved_mask)
|
||||
{
|
||||
int i;
|
||||
|
||||
writel(0, addr);
|
||||
for (i = 0; i < 32; i++)
|
||||
writel(GENMASK(i, 0), addr);
|
||||
writel(GENMASK(i, 0) & ~reserved_mask, addr);
|
||||
}
|
||||
|
||||
static int mt8186_scp_before_load(struct mtk_scp *scp)
|
||||
@ -393,7 +393,7 @@ static int mt8186_scp_before_load(struct mtk_scp *scp)
|
||||
writel(0x0, scp->reg_base + MT8183_SCP_CLK_DIV_SEL);
|
||||
|
||||
/* Turn on the power of SCP's SRAM before using it. Enable 1 block per time*/
|
||||
mt8192_power_on_sram(scp->reg_base + MT8183_SCP_SRAM_PDN);
|
||||
scp_sram_power_on(scp->reg_base + MT8183_SCP_SRAM_PDN, 0);
|
||||
|
||||
/* Initialize TCM before loading FW. */
|
||||
writel(0x0, scp->reg_base + MT8183_SCP_L1_SRAM_PD);
|
||||
@ -412,11 +412,32 @@ static int mt8192_scp_before_load(struct mtk_scp *scp)
|
||||
writel(1, scp->reg_base + MT8192_CORE0_SW_RSTN_SET);
|
||||
|
||||
/* enable SRAM clock */
|
||||
mt8192_power_on_sram(scp->reg_base + MT8192_L2TCM_SRAM_PD_0);
|
||||
mt8192_power_on_sram(scp->reg_base + MT8192_L2TCM_SRAM_PD_1);
|
||||
mt8192_power_on_sram(scp->reg_base + MT8192_L2TCM_SRAM_PD_2);
|
||||
mt8192_power_on_sram(scp->reg_base + MT8192_L1TCM_SRAM_PDN);
|
||||
mt8192_power_on_sram(scp->reg_base + MT8192_CPU0_SRAM_PD);
|
||||
scp_sram_power_on(scp->reg_base + MT8192_L2TCM_SRAM_PD_0, 0);
|
||||
scp_sram_power_on(scp->reg_base + MT8192_L2TCM_SRAM_PD_1, 0);
|
||||
scp_sram_power_on(scp->reg_base + MT8192_L2TCM_SRAM_PD_2, 0);
|
||||
scp_sram_power_on(scp->reg_base + MT8192_L1TCM_SRAM_PDN, 0);
|
||||
scp_sram_power_on(scp->reg_base + MT8192_CPU0_SRAM_PD, 0);
|
||||
|
||||
/* enable MPU for all memory regions */
|
||||
writel(0xff, scp->reg_base + MT8192_CORE0_MEM_ATT_PREDEF);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int mt8195_scp_before_load(struct mtk_scp *scp)
|
||||
{
|
||||
/* clear SPM interrupt, SCP2SPM_IPC_CLR */
|
||||
writel(0xff, scp->reg_base + MT8192_SCP2SPM_IPC_CLR);
|
||||
|
||||
writel(1, scp->reg_base + MT8192_CORE0_SW_RSTN_SET);
|
||||
|
||||
/* enable SRAM clock */
|
||||
scp_sram_power_on(scp->reg_base + MT8192_L2TCM_SRAM_PD_0, 0);
|
||||
scp_sram_power_on(scp->reg_base + MT8192_L2TCM_SRAM_PD_1, 0);
|
||||
scp_sram_power_on(scp->reg_base + MT8192_L2TCM_SRAM_PD_2, 0);
|
||||
scp_sram_power_on(scp->reg_base + MT8192_L1TCM_SRAM_PDN,
|
||||
MT8195_L1TCM_SRAM_PDN_RESERVED_RSI_BITS);
|
||||
scp_sram_power_on(scp->reg_base + MT8192_CPU0_SRAM_PD, 0);
|
||||
|
||||
/* enable MPU for all memory regions */
|
||||
writel(0xff, scp->reg_base + MT8192_CORE0_MEM_ATT_PREDEF);
|
||||
@ -572,11 +593,25 @@ static void mt8183_scp_stop(struct mtk_scp *scp)
|
||||
static void mt8192_scp_stop(struct mtk_scp *scp)
|
||||
{
|
||||
/* Disable SRAM clock */
|
||||
mt8192_power_off_sram(scp->reg_base + MT8192_L2TCM_SRAM_PD_0);
|
||||
mt8192_power_off_sram(scp->reg_base + MT8192_L2TCM_SRAM_PD_1);
|
||||
mt8192_power_off_sram(scp->reg_base + MT8192_L2TCM_SRAM_PD_2);
|
||||
mt8192_power_off_sram(scp->reg_base + MT8192_L1TCM_SRAM_PDN);
|
||||
mt8192_power_off_sram(scp->reg_base + MT8192_CPU0_SRAM_PD);
|
||||
scp_sram_power_off(scp->reg_base + MT8192_L2TCM_SRAM_PD_0, 0);
|
||||
scp_sram_power_off(scp->reg_base + MT8192_L2TCM_SRAM_PD_1, 0);
|
||||
scp_sram_power_off(scp->reg_base + MT8192_L2TCM_SRAM_PD_2, 0);
|
||||
scp_sram_power_off(scp->reg_base + MT8192_L1TCM_SRAM_PDN, 0);
|
||||
scp_sram_power_off(scp->reg_base + MT8192_CPU0_SRAM_PD, 0);
|
||||
|
||||
/* Disable SCP watchdog */
|
||||
writel(0, scp->reg_base + MT8192_CORE0_WDT_CFG);
|
||||
}
|
||||
|
||||
static void mt8195_scp_stop(struct mtk_scp *scp)
|
||||
{
|
||||
/* Disable SRAM clock */
|
||||
scp_sram_power_off(scp->reg_base + MT8192_L2TCM_SRAM_PD_0, 0);
|
||||
scp_sram_power_off(scp->reg_base + MT8192_L2TCM_SRAM_PD_1, 0);
|
||||
scp_sram_power_off(scp->reg_base + MT8192_L2TCM_SRAM_PD_2, 0);
|
||||
scp_sram_power_off(scp->reg_base + MT8192_L1TCM_SRAM_PDN,
|
||||
MT8195_L1TCM_SRAM_PDN_RESERVED_RSI_BITS);
|
||||
scp_sram_power_off(scp->reg_base + MT8192_CPU0_SRAM_PD, 0);
|
||||
|
||||
/* Disable SCP watchdog */
|
||||
writel(0, scp->reg_base + MT8192_CORE0_WDT_CFG);
|
||||
@ -774,9 +809,13 @@ static int scp_probe(struct platform_device *pdev)
|
||||
struct mtk_scp *scp;
|
||||
struct rproc *rproc;
|
||||
struct resource *res;
|
||||
char *fw_name = "scp.img";
|
||||
const char *fw_name = "scp.img";
|
||||
int ret, i;
|
||||
|
||||
ret = rproc_of_parse_firmware(dev, 0, &fw_name);
|
||||
if (ret < 0 && ret != -EINVAL)
|
||||
return ret;
|
||||
|
||||
rproc = devm_rproc_alloc(dev, np->name, &scp_ops, fw_name, sizeof(*scp));
|
||||
if (!rproc)
|
||||
return dev_err_probe(dev, -ENOMEM, "unable to allocate remoteproc\n");
|
||||
@ -877,7 +916,6 @@ static int scp_remove(struct platform_device *pdev)
|
||||
for (i = 0; i < SCP_IPI_MAX; i++)
|
||||
mutex_destroy(&scp->ipi_desc[i].lock);
|
||||
mutex_destroy(&scp->send_lock);
|
||||
rproc_free(scp->rproc);
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -922,11 +960,11 @@ static const struct mtk_scp_of_data mt8192_of_data = {
|
||||
|
||||
static const struct mtk_scp_of_data mt8195_of_data = {
|
||||
.scp_clk_get = mt8195_scp_clk_get,
|
||||
.scp_before_load = mt8192_scp_before_load,
|
||||
.scp_before_load = mt8195_scp_before_load,
|
||||
.scp_irq_handler = mt8192_scp_irq_handler,
|
||||
.scp_reset_assert = mt8192_scp_reset_assert,
|
||||
.scp_reset_deassert = mt8192_scp_reset_deassert,
|
||||
.scp_stop = mt8192_scp_stop,
|
||||
.scp_stop = mt8195_scp_stop,
|
||||
.scp_da_to_va = mt8192_scp_da_to_va,
|
||||
.host_to_scp_reg = MT8192_GIPC_IN_SET,
|
||||
.host_to_scp_int_bit = MT8192_HOST_IPC_INT_BIT,
|
||||
|
@ -704,6 +704,36 @@ static const struct adsp_data sm8250_cdsp_resource = {
|
||||
.ssctl_id = 0x17,
|
||||
};
|
||||
|
||||
static const struct adsp_data sc8280xp_nsp0_resource = {
|
||||
.crash_reason_smem = 601,
|
||||
.firmware_name = "cdsp.mdt",
|
||||
.pas_id = 18,
|
||||
.has_aggre2_clk = false,
|
||||
.auto_boot = true,
|
||||
.proxy_pd_names = (char*[]){
|
||||
"nsp",
|
||||
NULL
|
||||
},
|
||||
.ssr_name = "cdsp0",
|
||||
.sysmon_name = "cdsp",
|
||||
.ssctl_id = 0x17,
|
||||
};
|
||||
|
||||
static const struct adsp_data sc8280xp_nsp1_resource = {
|
||||
.crash_reason_smem = 633,
|
||||
.firmware_name = "cdsp.mdt",
|
||||
.pas_id = 30,
|
||||
.has_aggre2_clk = false,
|
||||
.auto_boot = true,
|
||||
.proxy_pd_names = (char*[]){
|
||||
"nsp",
|
||||
NULL
|
||||
},
|
||||
.ssr_name = "cdsp1",
|
||||
.sysmon_name = "cdsp1",
|
||||
.ssctl_id = 0x20,
|
||||
};
|
||||
|
||||
static const struct adsp_data sm8350_cdsp_resource = {
|
||||
.crash_reason_smem = 601,
|
||||
.firmware_name = "cdsp.mdt",
|
||||
@ -848,6 +878,7 @@ static const struct adsp_data sdx55_mpss_resource = {
|
||||
};
|
||||
|
||||
static const struct of_device_id adsp_of_match[] = {
|
||||
{ .compatible = "qcom,msm8226-adsp-pil", .data = &adsp_resource_init},
|
||||
{ .compatible = "qcom,msm8974-adsp-pil", .data = &adsp_resource_init},
|
||||
{ .compatible = "qcom,msm8996-adsp-pil", .data = &msm8996_adsp_resource},
|
||||
{ .compatible = "qcom,msm8996-slpi-pil", .data = &slpi_resource_init},
|
||||
@ -861,6 +892,9 @@ static const struct of_device_id adsp_of_match[] = {
|
||||
{ .compatible = "qcom,sc8180x-adsp-pas", .data = &sm8150_adsp_resource},
|
||||
{ .compatible = "qcom,sc8180x-cdsp-pas", .data = &sm8150_cdsp_resource},
|
||||
{ .compatible = "qcom,sc8180x-mpss-pas", .data = &sc8180x_mpss_resource},
|
||||
{ .compatible = "qcom,sc8280xp-adsp-pas", .data = &sm8250_adsp_resource},
|
||||
{ .compatible = "qcom,sc8280xp-nsp0-pas", .data = &sc8280xp_nsp0_resource},
|
||||
{ .compatible = "qcom,sc8280xp-nsp1-pas", .data = &sc8280xp_nsp1_resource},
|
||||
{ .compatible = "qcom,sdm660-adsp-pas", .data = &adsp_resource_init},
|
||||
{ .compatible = "qcom,sdm845-adsp-pas", .data = &sdm845_adsp_resource_init},
|
||||
{ .compatible = "qcom,sdm845-cdsp-pas", .data = &sdm845_cdsp_resource_init},
|
||||
|
@ -32,21 +32,10 @@ static ssize_t rproc_cdev_write(struct file *filp, const char __user *buf, size_
|
||||
return -EFAULT;
|
||||
|
||||
if (!strncmp(cmd, "start", len)) {
|
||||
if (rproc->state == RPROC_RUNNING ||
|
||||
rproc->state == RPROC_ATTACHED)
|
||||
return -EBUSY;
|
||||
|
||||
ret = rproc_boot(rproc);
|
||||
} else if (!strncmp(cmd, "stop", len)) {
|
||||
if (rproc->state != RPROC_RUNNING &&
|
||||
rproc->state != RPROC_ATTACHED)
|
||||
return -EINVAL;
|
||||
|
||||
ret = rproc_shutdown(rproc);
|
||||
} else if (!strncmp(cmd, "detach", len)) {
|
||||
if (rproc->state != RPROC_ATTACHED)
|
||||
return -EINVAL;
|
||||
|
||||
ret = rproc_detach(rproc);
|
||||
} else {
|
||||
dev_err(&rproc->dev, "Unrecognized option\n");
|
||||
|
@ -684,10 +684,6 @@ static int rproc_handle_trace(struct rproc *rproc, void *ptr,
|
||||
|
||||
/* create the debugfs entry */
|
||||
trace->tfile = rproc_create_trace_file(name, rproc, trace);
|
||||
if (!trace->tfile) {
|
||||
kfree(trace);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
list_add_tail(&trace->node, &rproc->traces);
|
||||
|
||||
@ -2075,6 +2071,12 @@ int rproc_shutdown(struct rproc *rproc)
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (rproc->state != RPROC_RUNNING &&
|
||||
rproc->state != RPROC_ATTACHED) {
|
||||
ret = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
/* if the remote proc is still needed, bail out */
|
||||
if (!atomic_dec_and_test(&rproc->power))
|
||||
goto out;
|
||||
@ -2134,6 +2136,11 @@ int rproc_detach(struct rproc *rproc)
|
||||
return ret;
|
||||
}
|
||||
|
||||
if (rproc->state != RPROC_ATTACHED) {
|
||||
ret = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
/* if the remote proc is still needed, bail out */
|
||||
if (!atomic_dec_and_test(&rproc->power)) {
|
||||
ret = 0;
|
||||
|
@ -386,16 +386,8 @@ void rproc_remove_trace_file(struct dentry *tfile)
|
||||
struct dentry *rproc_create_trace_file(const char *name, struct rproc *rproc,
|
||||
struct rproc_debug_trace *trace)
|
||||
{
|
||||
struct dentry *tfile;
|
||||
|
||||
tfile = debugfs_create_file(name, 0400, rproc->dbg_dir, trace,
|
||||
return debugfs_create_file(name, 0400, rproc->dbg_dir, trace,
|
||||
&trace_rproc_ops);
|
||||
if (!tfile) {
|
||||
dev_err(&rproc->dev, "failed to create debugfs trace entry\n");
|
||||
return NULL;
|
||||
}
|
||||
|
||||
return tfile;
|
||||
}
|
||||
|
||||
void rproc_delete_debug_dir(struct rproc *rproc)
|
||||
@ -411,8 +403,6 @@ void rproc_create_debug_dir(struct rproc *rproc)
|
||||
return;
|
||||
|
||||
rproc->dbg_dir = debugfs_create_dir(dev_name(dev), rproc_dbg);
|
||||
if (!rproc->dbg_dir)
|
||||
return;
|
||||
|
||||
debugfs_create_file("name", 0400, rproc->dbg_dir,
|
||||
rproc, &rproc_name_ops);
|
||||
@ -430,11 +420,8 @@ void rproc_create_debug_dir(struct rproc *rproc)
|
||||
|
||||
void __init rproc_init_debugfs(void)
|
||||
{
|
||||
if (debugfs_initialized()) {
|
||||
if (debugfs_initialized())
|
||||
rproc_dbg = debugfs_create_dir(KBUILD_MODNAME, NULL);
|
||||
if (!rproc_dbg)
|
||||
pr_err("can't create debugfs dir\n");
|
||||
}
|
||||
}
|
||||
|
||||
void __exit rproc_exit_debugfs(void)
|
||||
|
@ -181,7 +181,7 @@ int rproc_elf_load_segments(struct rproc *rproc, const struct firmware *fw)
|
||||
bool is_iomem = false;
|
||||
void *ptr;
|
||||
|
||||
if (type != PT_LOAD)
|
||||
if (type != PT_LOAD || !memsz)
|
||||
continue;
|
||||
|
||||
dev_dbg(dev, "phdr: type %d da 0x%llx memsz 0x%llx filesz 0x%llx\n",
|
||||
|
@ -194,23 +194,12 @@ static ssize_t state_store(struct device *dev,
|
||||
int ret = 0;
|
||||
|
||||
if (sysfs_streq(buf, "start")) {
|
||||
if (rproc->state == RPROC_RUNNING ||
|
||||
rproc->state == RPROC_ATTACHED)
|
||||
return -EBUSY;
|
||||
|
||||
ret = rproc_boot(rproc);
|
||||
if (ret)
|
||||
dev_err(&rproc->dev, "Boot failed: %d\n", ret);
|
||||
} else if (sysfs_streq(buf, "stop")) {
|
||||
if (rproc->state != RPROC_RUNNING &&
|
||||
rproc->state != RPROC_ATTACHED)
|
||||
return -EINVAL;
|
||||
|
||||
ret = rproc_shutdown(rproc);
|
||||
} else if (sysfs_streq(buf, "detach")) {
|
||||
if (rproc->state != RPROC_ATTACHED)
|
||||
return -EINVAL;
|
||||
|
||||
ret = rproc_detach(rproc);
|
||||
} else {
|
||||
dev_err(&rproc->dev, "Unrecognised option: %s\n", buf);
|
||||
|
Loading…
Reference in New Issue
Block a user