On Mon, Jun 22, 2026 at 09:31:14AM +0530, Sailesh Nandanavanam wrote:
> qcom_pas_load() acquires a reference to pas->dtb_firmware via
> request_firmware() whenever the platform defines dtb_pas_id. This
> reference is only released on the error path inside qcom_pas_load()
> itself, when qcom_mdt_pas_load() fails.
> 
> On the success path, pas->dtb_firmware is never released by
> qcom_pas_start(), qcom_pas_stop(), qcom_pas_unprepare(), or
> qcom_pas_remove(). Since qcom_pas_load() runs on every load/reload
> cycle, including subsystem-restart recovery, this leaks a firmware
> reference on every successful boot of remoteprocs that use a DTB
> co-firmware image (e.g. ADSP/CDSP/MPSS on SM8550, SM8650, SM8750,
> X1E80100).
> 
> A second, narrower leak window exists if qcom_pas_start() itself
> fails after the DTB carveout has been mapped: the remoteproc core
> does not invoke .stop for a remoteproc that failed to start, so
> qcom_pas_stop() never runs to release the reference in that case.

> 
> Release pas->dtb_firmware in both qcom_pas_stop() and at the
> unmap_dtb_carveout label in qcom_pas_start(), alongside the existing
> teardown of the DTB carveout mapping, and clear the pointer to avoid
> holding a stale reference into the next load cycle.
> 
> Fixes: 29814986b82e ("remoteproc: qcom_q6v5_pas: add support for dtb 
> co-firmware loading")
>

Remove the line feed from here..

> Signed-off-by: Sailesh Nandanavanam <[email protected]>
> ---
>  drivers/remoteproc/qcom_q6v5_pas.c | 7 ++++++-
>  1 file changed, 6 insertions(+), 1 deletion(-)
> 
> diff --git a/drivers/remoteproc/qcom_q6v5_pas.c 
> b/drivers/remoteproc/qcom_q6v5_pas.c
> index da27d1d3c9da..040aabbe5860 100644
> --- a/drivers/remoteproc/qcom_q6v5_pas.c
> +++ b/drivers/remoteproc/qcom_q6v5_pas.c
> @@ -360,8 +360,11 @@ static int qcom_pas_start(struct rproc *rproc)
>               qcom_scm_pas_metadata_release(pas->dtb_pas_ctx);
>  
>  unmap_dtb_carveout:
> -     if (pas->dtb_pas_id)
> +     if (pas->dtb_pas_id) {
>               qcom_pas_unmap_carveout(rproc, pas->dtb_mem_phys, 
> pas->dtb_mem_size);
> +             release_firmware(pas->dtb_firmware);

This is still not solving the issue. What if you have failed just after
the first function qcom_q6v5_prepare() in qcom_pas_start()?

would it be better to move dtb_firmware loading to qcom_pas_start() ?


> +             pas->dtb_firmware = NULL;
> +     }
>  disable_px_supply:
>       if (pas->px_supply)
>               regulator_disable(pas->px_supply);
> @@ -419,6 +422,8 @@ static int qcom_pas_stop(struct rproc *rproc)
>                       dev_err(pas->dev, "failed to shutdown dtb: %d\n", ret);
>  
>               qcom_pas_unmap_carveout(rproc, pas->dtb_mem_phys, 
> pas->dtb_mem_size);
> +             release_firmware(pas->dtb_firmware);
> +             pas->dtb_firmware = NULL;
>       }
>  
>       qcom_pas_unmap_carveout(rproc, pas->mem_phys, pas->mem_size);
> -- 
> 2.34.1
> 

-- 
-Mukesh Ojha

Reply via email to