We add support for reporting 'fuel-gauge' NVDIMM metric via
PAPR_PDSM_HEALTH pdsm payload. 'fuel-gauge' metric indicates the usage
life remaining of a papr-scm compatible NVDIMM. PHYP exposes this
metric via the H_SCM_PERFORMANCE_STATS.

The metric value is returned from the pdsm by extending the return
payload 'struct nd_papr_pdsm_health' without breaking the ABI. A new
field 'dimm_fuel_gauge' to hold the metric value is introduced at the
end of the payload struct and its presence is indicated by by
extension flag PDSM_DIMM_HEALTH_RUN_GAUGE_VALID.

The patch introduces a new function papr_pdsm_fuel_gauge() that is
called from papr_pdsm_health(). If fetching NVDIMM performance stats
is supported then 'papr_pdsm_fuel_gauge()' allocated an output buffer
large enough to hold the performance stat and passes it to
drc_pmem_query_stats() that issues the HCALL to PHYP. The return value
of the stat is then populated in the 'struct
nd_papr_pdsm_health.dimm_fuel_gauge' field with extension flag
'PDSM_DIMM_HEALTH_RUN_GAUGE_VALID' set in 'struct
nd_papr_pdsm_health.extension_flags'

Signed-off-by: Vaibhav Jain <vaib...@linux.ibm.com>
---
 arch/powerpc/include/uapi/asm/papr_pdsm.h |  9 +++++
 arch/powerpc/platforms/pseries/papr_scm.c | 47 +++++++++++++++++++++++
 2 files changed, 56 insertions(+)

diff --git a/arch/powerpc/include/uapi/asm/papr_pdsm.h 
b/arch/powerpc/include/uapi/asm/papr_pdsm.h
index 9ccecc1d6840..50ef95e2f5b1 100644
--- a/arch/powerpc/include/uapi/asm/papr_pdsm.h
+++ b/arch/powerpc/include/uapi/asm/papr_pdsm.h
@@ -72,6 +72,11 @@
 #define PAPR_PDSM_DIMM_CRITICAL      2
 #define PAPR_PDSM_DIMM_FATAL         3
 
+/* struct nd_papr_pdsm_health.extension_flags field flags */
+
+/* Indicate that the 'dimm_fuel_gauge' field is valid */
+#define PDSM_DIMM_HEALTH_RUN_GAUGE_VALID 1
+
 /*
  * Struct exchanged between kernel & ndctl in for PAPR_PDSM_HEALTH
  * Various flags indicate the health status of the dimm.
@@ -84,6 +89,7 @@
  * dimm_locked         : Contents of the dimm cant be modified until CEC reboot
  * dimm_encrypted      : Contents of dimm are encrypted.
  * dimm_health         : Dimm health indicator. One of PAPR_PDSM_DIMM_XXXX
+ * dimm_fuel_gauge     : Life remaining of DIMM as a percentage from 0-100
  */
 struct nd_papr_pdsm_health {
        union {
@@ -96,6 +102,9 @@ struct nd_papr_pdsm_health {
                        __u8 dimm_locked;
                        __u8 dimm_encrypted;
                        __u16 dimm_health;
+
+                       /* Extension flag PDSM_DIMM_HEALTH_RUN_GAUGE_VALID */
+                       __u16 dimm_fuel_gauge;
                };
                __u8 buf[ND_PDSM_PAYLOAD_MAX_SIZE];
        };
diff --git a/arch/powerpc/platforms/pseries/papr_scm.c 
b/arch/powerpc/platforms/pseries/papr_scm.c
index cb3f9acc325b..39527cd38d9c 100644
--- a/arch/powerpc/platforms/pseries/papr_scm.c
+++ b/arch/powerpc/platforms/pseries/papr_scm.c
@@ -506,6 +506,45 @@ static int is_cmd_valid(struct nvdimm *nvdimm, unsigned 
int cmd, void *buf,
        return 0;
 }
 
+static int papr_pdsm_fuel_gauge(struct papr_scm_priv *p,
+                               union nd_pdsm_payload *payload)
+{
+       int rc, size;
+       struct papr_scm_perf_stat *stat;
+       struct papr_scm_perf_stats *stats;
+
+       /* Silently fail if fetching performance metrics isn't  supported */
+       if (!p->len_stat_buffer)
+               return 0;
+
+       /* Allocate request buffer enough to hold single performance stat */
+       size = sizeof(struct papr_scm_perf_stats) +
+               sizeof(struct papr_scm_perf_stat);
+
+       stats = kzalloc(size, GFP_KERNEL);
+       if (!stats)
+               return -ENOMEM;
+
+       stat = &stats->scm_statistic[0];
+       memcpy(&stat->statistic_id, "MemLife ", sizeof(stat->statistic_id));
+       stat->statistic_value = 0;
+
+       /* Fetch the fuel gauge and populate it in payload */
+       rc = drc_pmem_query_stats(p, stats, size, 1, NULL);
+       if (!rc) {
+               dev_dbg(&p->pdev->dev,
+                       "Fetched fuel-gauge %llu", stat->statistic_value);
+               payload->health.extension_flags |=
+                       PDSM_DIMM_HEALTH_RUN_GAUGE_VALID;
+               payload->health.dimm_fuel_gauge = stat->statistic_value;
+
+               rc = sizeof(struct nd_papr_pdsm_health);
+       }
+
+       kfree(stats);
+       return rc;
+}
+
 /* Fetch the DIMM health info and populate it in provided package. */
 static int papr_pdsm_health(struct papr_scm_priv *p,
                            union nd_pdsm_payload *payload)
@@ -546,6 +585,14 @@ static int papr_pdsm_health(struct papr_scm_priv *p,
 
        /* struct populated hence can release the mutex now */
        mutex_unlock(&p->health_mutex);
+
+       /* Populate the fuel gauge meter in the payload */
+       rc = papr_pdsm_fuel_gauge(p, payload);
+
+       /* Error fetching fuel gauge is not fatal */
+       if (rc < 0)
+               dev_dbg(&p->pdev->dev, "Err(%d) fetching fuel gauge\n", rc);
+
        rc = sizeof(struct nd_papr_pdsm_health);
 
 out:
-- 
2.26.2

Reply via email to