Commit a376c10d authored by Yogesh Lal's avatar Yogesh Lal Committed by Bjorn Andersson

remoteproc: qcom: pas: Adjust the phys addr wrt the mem region

The minidump table in the toc contains physical addresses that may lie
before the physical address of the first elf segment in relocatable
images. This change adds a custom dump function for minidumps which
calculates the offset into the carveout region using the start of
the physical address instead of the start of the first elf segment.
Signed-off-by: default avatarYogesh Lal <quic_ylal@quicinc.com>
Reviewed-by: default avatarSibi Sankar <quic_sibis@quicinc.com>
Signed-off-by: default avatarBjorn Andersson <andersson@kernel.org>
Link: https://lore.kernel.org/r/1667409129-6254-1-git-send-email-quic_ylal@quicinc.com
parent 2554dd0a
...@@ -101,7 +101,9 @@ static void qcom_minidump_cleanup(struct rproc *rproc) ...@@ -101,7 +101,9 @@ static void qcom_minidump_cleanup(struct rproc *rproc)
} }
} }
static int qcom_add_minidump_segments(struct rproc *rproc, struct minidump_subsystem *subsystem) static int qcom_add_minidump_segments(struct rproc *rproc, struct minidump_subsystem *subsystem,
void (*rproc_dumpfn_t)(struct rproc *rproc, struct rproc_dump_segment *segment,
void *dest, size_t offset, size_t size))
{ {
struct minidump_region __iomem *ptr; struct minidump_region __iomem *ptr;
struct minidump_region region; struct minidump_region region;
...@@ -131,7 +133,7 @@ static int qcom_add_minidump_segments(struct rproc *rproc, struct minidump_subsy ...@@ -131,7 +133,7 @@ static int qcom_add_minidump_segments(struct rproc *rproc, struct minidump_subsy
} }
da = le64_to_cpu(region.address); da = le64_to_cpu(region.address);
size = le64_to_cpu(region.size); size = le64_to_cpu(region.size);
rproc_coredump_add_custom_segment(rproc, da, size, NULL, name); rproc_coredump_add_custom_segment(rproc, da, size, rproc_dumpfn_t, name);
} }
} }
...@@ -139,7 +141,10 @@ static int qcom_add_minidump_segments(struct rproc *rproc, struct minidump_subsy ...@@ -139,7 +141,10 @@ static int qcom_add_minidump_segments(struct rproc *rproc, struct minidump_subsy
return 0; return 0;
} }
void qcom_minidump(struct rproc *rproc, unsigned int minidump_id) void qcom_minidump(struct rproc *rproc, unsigned int minidump_id,
void (*rproc_dumpfn_t)(struct rproc *rproc,
struct rproc_dump_segment *segment, void *dest, size_t offset,
size_t size))
{ {
int ret; int ret;
struct minidump_subsystem *subsystem; struct minidump_subsystem *subsystem;
...@@ -169,7 +174,7 @@ void qcom_minidump(struct rproc *rproc, unsigned int minidump_id) ...@@ -169,7 +174,7 @@ void qcom_minidump(struct rproc *rproc, unsigned int minidump_id)
return; return;
} }
ret = qcom_add_minidump_segments(rproc, subsystem); ret = qcom_add_minidump_segments(rproc, subsystem, rproc_dumpfn_t);
if (ret) { if (ret) {
dev_err(&rproc->dev, "Failed with error: %d while adding minidump entries\n", ret); dev_err(&rproc->dev, "Failed with error: %d while adding minidump entries\n", ret);
goto clean_minidump; goto clean_minidump;
......
...@@ -33,7 +33,10 @@ struct qcom_rproc_ssr { ...@@ -33,7 +33,10 @@ struct qcom_rproc_ssr {
struct qcom_ssr_subsystem *info; struct qcom_ssr_subsystem *info;
}; };
void qcom_minidump(struct rproc *rproc, unsigned int minidump_id); void qcom_minidump(struct rproc *rproc, unsigned int minidump_id,
void (*rproc_dumpfn_t)(struct rproc *rproc,
struct rproc_dump_segment *segment, void *dest, size_t offset,
size_t size));
void qcom_add_glink_subdev(struct rproc *rproc, struct qcom_rproc_glink *glink, void qcom_add_glink_subdev(struct rproc *rproc, struct qcom_rproc_glink *glink,
const char *ssr_name); const char *ssr_name);
......
...@@ -105,6 +105,24 @@ struct qcom_adsp { ...@@ -105,6 +105,24 @@ struct qcom_adsp {
struct qcom_scm_pas_metadata dtb_pas_metadata; struct qcom_scm_pas_metadata dtb_pas_metadata;
}; };
void adsp_segment_dump(struct rproc *rproc, struct rproc_dump_segment *segment,
void *dest, size_t offset, size_t size)
{
struct qcom_adsp *adsp = rproc->priv;
int total_offset;
total_offset = segment->da + segment->offset + offset - adsp->mem_phys;
if (total_offset < 0 || total_offset + size > adsp->mem_size) {
dev_err(adsp->dev,
"invalid copy request for segment %pad with offset %zu and size %zu)\n",
&segment->da, offset, size);
memset(dest, 0xff, size);
return;
}
memcpy_fromio(dest, adsp->mem_region + total_offset, size);
}
static void adsp_minidump(struct rproc *rproc) static void adsp_minidump(struct rproc *rproc)
{ {
struct qcom_adsp *adsp = rproc->priv; struct qcom_adsp *adsp = rproc->priv;
...@@ -112,7 +130,7 @@ static void adsp_minidump(struct rproc *rproc) ...@@ -112,7 +130,7 @@ static void adsp_minidump(struct rproc *rproc)
if (rproc->dump_conf == RPROC_COREDUMP_DISABLED) if (rproc->dump_conf == RPROC_COREDUMP_DISABLED)
return; return;
qcom_minidump(rproc, adsp->minidump_id); qcom_minidump(rproc, adsp->minidump_id, adsp_segment_dump);
} }
static int adsp_pds_enable(struct qcom_adsp *adsp, struct device **pds, static int adsp_pds_enable(struct qcom_adsp *adsp, struct device **pds,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment