On Tue, May 19, 2026 at 12:24:23AM -0700, Jingyi Wang wrote:
> Subsystems can be brought out of reset by entities such as bootloaders.
> As the irq enablement could be later than subsystem bring up, the state
> of subsystem should be checked by reading SMP2P bits.
> 
> A new qcom_pas_attach() function is introduced. if a crash state is
> detected for the subsystem, rproc_report_crash() is called. If the ready
> state is detected, it will be marked as "attached", otherwise it could
> be the early boot feature is not supported by other entities. In this
> case, the state will be marked as RPROC_OFFLINE so that the PAS driver
> can load the firmware and start the remoteproc.
> 
> Co-developed-by: Gokul Krishna Krishnakumar 
> <[email protected]>
> Signed-off-by: Gokul Krishna Krishnakumar 
> <[email protected]>
> Signed-off-by: Jingyi Wang <[email protected]>

Thank you for the work, Jingyi!

Tested-by: Shawn Guo <[email protected]> # Nord ADSP

A couple of minor nits below.

> ---
>  drivers/remoteproc/qcom_q6v5_pas.c | 58 
> ++++++++++++++++++++++++++++++++++++++
>  1 file changed, 58 insertions(+)
> 
> diff --git a/drivers/remoteproc/qcom_q6v5_pas.c 
> b/drivers/remoteproc/qcom_q6v5_pas.c
> index da27d1d3c9da..ac2a00aacd2e 100644
> --- a/drivers/remoteproc/qcom_q6v5_pas.c
> +++ b/drivers/remoteproc/qcom_q6v5_pas.c
> @@ -60,6 +60,7 @@ struct qcom_pas_data {
>       int region_assign_count;
>       bool region_assign_shared;
>       int region_assign_vmid;
> +     bool early_boot;
>  };
>  
>  struct qcom_pas {
> @@ -510,6 +511,57 @@ static unsigned long qcom_pas_panic(struct rproc *rproc)
>       return qcom_q6v5_panic(&pas->q6v5);
>  }
>  
> +static int qcom_pas_attach(struct rproc *rproc)
> +{
> +     int ret;
> +     struct qcom_pas *pas = rproc->priv;
> +     bool ready_state;
> +     bool crash_state;

Move "int ret;" line here to get reverse xmas tree.

> +
> +     pas->q6v5.handover_issued = true;
> +     enable_irq(pas->q6v5.handover_irq);
> +
> +     pas->q6v5.running = true;
> +     ret = irq_get_irqchip_state(pas->q6v5.fatal_irq,
> +                                 IRQCHIP_STATE_LINE_LEVEL, &crash_state);
> +

Unneeded newline.

> +     if (ret)
> +             goto disable_running;
> +
> +     if (crash_state) {
> +             dev_err(pas->dev, "Subsystem has crashed before driver 
> probe\n");
> +             rproc_report_crash(rproc, RPROC_FATAL_ERROR);
> +             ret = -EINVAL;
> +             goto disable_running;
> +     }
> +
> +     ret = irq_get_irqchip_state(pas->q6v5.ready_irq,
> +                                 IRQCHIP_STATE_LINE_LEVEL, &ready_state);
> +

Ditto

Shawn

> +     if (ret)
> +             goto disable_running;
> +
> +     if (unlikely(!ready_state)) {
> +             /*
> +              * The bootloader may not support early boot, mark the state as
> +              * RPROC_OFFLINE so that the PAS driver can load the firmware 
> and
> +              * start the remoteproc.
> +              */
> +             dev_err(pas->dev, "Failed to get subsystem ready interrupt\n");
> +             pas->rproc->state = RPROC_OFFLINE;
> +             disable_irq(pas->q6v5.handover_irq);
> +             ret = -EINVAL;
> +             goto disable_running;
> +     }
> +
> +     return 0;
> +
> +disable_running:
> +     pas->q6v5.running = false;
> +
> +     return ret;
> +}
> +
>  static const struct rproc_ops qcom_pas_ops = {
>       .unprepare = qcom_pas_unprepare,
>       .start = qcom_pas_start,
> @@ -518,6 +570,7 @@ static const struct rproc_ops qcom_pas_ops = {
>       .parse_fw = qcom_pas_parse_firmware,
>       .load = qcom_pas_load,
>       .panic = qcom_pas_panic,
> +     .attach = qcom_pas_attach,
>  };
>  
>  static const struct rproc_ops qcom_pas_minidump_ops = {
> @@ -529,6 +582,7 @@ static const struct rproc_ops qcom_pas_minidump_ops = {
>       .load = qcom_pas_load,
>       .panic = qcom_pas_panic,
>       .coredump = qcom_pas_minidump,
> +     .attach = qcom_pas_attach,
>  };
>  
>  static int qcom_pas_init_clock(struct qcom_pas *pas)
> @@ -855,6 +909,10 @@ static int qcom_pas_probe(struct platform_device *pdev)
>  
>       pas->pas_ctx->use_tzmem = rproc->has_iommu;
>       pas->dtb_pas_ctx->use_tzmem = rproc->has_iommu;
> +
> +     if (desc->early_boot)
> +             pas->rproc->state = RPROC_DETACHED;
> +
>       ret = rproc_add(rproc);
>       if (ret)
>               goto remove_ssr_sysmon;
> 
> -- 
> 2.34.1
> 
> 

Reply via email to