@@ -137,6 +137,21 @@ enum pstore_type_id pstore_name_to_type(const char *name)
}
EXPORT_SYMBOL_GPL(pstore_name_to_type);
+int pstore_region_defined(struct pstore_record *record,
+ void **virt, phys_addr_t *phys,
+ size_t *size, unsigned int *max_dump_cnt)
+{
+ if (!psinfo)
+ return -EINVAL;
+
+ record->psi = psinfo;
+
+ return psinfo->region_info ?
+ psinfo->region_info(record, virt, phys, size, max_dump_cnt) :
+ -EINVAL;
+}
+EXPORT_SYMBOL_GPL(pstore_region_defined);
+
static void pstore_timer_kick(void)
{
if (pstore_update_ms < 0)
@@ -436,6 +436,47 @@ static int ramoops_pstore_erase(struct pstore_record *record)
return 0;
}
+static int ramoops_region_info(struct pstore_record *record,
+ void **virt, phys_addr_t *phys,
+ size_t *size, unsigned int *max_dump_cnt)
+{
+ struct ramoops_context *cxt = record->psi->data;
+ struct persistent_ram_zone *prz;
+
+ switch (record->type) {
+ case PSTORE_TYPE_DMESG:
+ if (record->id >= cxt->max_dump_cnt)
+ return -EINVAL;
+ prz = cxt->dprzs[record->id];
+ *max_dump_cnt = cxt->max_dump_cnt;
+ break;
+ case PSTORE_TYPE_CONSOLE:
+ if (!cxt->console_size)
+ return -EINVAL;
+ prz = cxt->cprz;
+ break;
+ case PSTORE_TYPE_FTRACE:
+ if (record->id >= cxt->max_ftrace_cnt)
+ return -EINVAL;
+ prz = cxt->fprzs[record->id];
+ *max_dump_cnt = cxt->max_ftrace_cnt;
+ break;
+ case PSTORE_TYPE_PMSG:
+ if (!cxt->pmsg_size)
+ return -EINVAL;
+ prz = cxt->mprz;
+ break;
+ default:
+ return -EINVAL;
+ }
+
+ *virt = prz->vaddr;
+ *phys = prz->paddr;
+ *size = prz->size;
+
+ return 0;
+}
+
static struct ramoops_context oops_cxt = {
.pstore = {
.owner = THIS_MODULE,
@@ -445,6 +486,7 @@ static struct ramoops_context oops_cxt = {
.write = ramoops_pstore_write,
.write_user = ramoops_pstore_write_user,
.erase = ramoops_pstore_erase,
+ .region_info = ramoops_region_info,
},
};
@@ -199,6 +199,9 @@ struct pstore_info {
int (*write_user)(struct pstore_record *record,
const char __user *buf);
int (*erase)(struct pstore_record *record);
+ int (*region_info)(struct pstore_record *record,
+ void **virt, phys_addr_t *phys,
+ size_t *size, unsigned int *max_dump_cnt);
};
/* Supported frontends */
@@ -230,6 +233,9 @@ struct pstore_ftrace_record {
#define TS_CPU_SHIFT 8
#define TS_CPU_MASK (BIT(TS_CPU_SHIFT) - 1)
+int pstore_region_defined(struct pstore_record *record,
+ void **virt, phys_addr_t *phys,
+ size_t *size, unsigned int *max_dump_cnt);
/*
* If CPU number can be stored in IP, store it there, otherwise store it in
* the time stamp. This means more timestamp resolution is available when
There are users like Qualcomm minidump which is interested in knowing the pstore frontend addresses and sizes from the backend (ram) to be able to register it with firmware to finally collect them during crash for debugging. Signed-off-by: Mukesh Ojha <quic_mojha@quicinc.com> --- fs/pstore/platform.c | 15 +++++++++++++++ fs/pstore/ram.c | 42 ++++++++++++++++++++++++++++++++++++++++++ include/linux/pstore.h | 6 ++++++ 3 files changed, 63 insertions(+)