Commit 4aa92c86 authored by Robert Richter's avatar Robert Richter Committed by Borislav Petkov

EDAC/mc: Remove per layer counters

Looking at how mci->{ue,ce}_per_layer[EDAC_MAX_LAYERS] is used, it
turns out that only the leaves in the memory hierarchy are consumed
(in sysfs), but not the intermediate layers, e.g.:

  count = dimm->mci->ce_per_layer[dimm->mci->n_layers-1][dimm->idx];

These unused counters only add complexity, remove them. The error
counter values are directly stored in struct dimm_info now.
Signed-off-by: default avatarRobert Richter <rrichter@marvell.com>
Signed-off-by: default avatarBorislav Petkov <bp@suse.de>
Acked-by: default avatarAristeu Rozanski <aris@redhat.com>
Link: https://lkml.kernel.org/r/20200123090210.26933-11-rrichter@marvell.com
parent 1853ee72
...@@ -451,11 +451,9 @@ struct mem_ctl_info *edac_mc_alloc(unsigned int mc_num, ...@@ -451,11 +451,9 @@ struct mem_ctl_info *edac_mc_alloc(unsigned int mc_num,
{ {
struct mem_ctl_info *mci; struct mem_ctl_info *mci;
struct edac_mc_layer *layer; struct edac_mc_layer *layer;
u32 *ce_per_layer[EDAC_MAX_LAYERS], *ue_per_layer[EDAC_MAX_LAYERS]; unsigned int idx, size, tot_dimms = 1;
unsigned int idx, size, tot_dimms = 1, count = 1; unsigned int tot_csrows = 1, tot_channels = 1;
unsigned int tot_csrows = 1, tot_channels = 1, tot_errcount = 0;
void *pvt, *ptr = NULL; void *pvt, *ptr = NULL;
int i;
bool per_rank = false; bool per_rank = false;
if (WARN_ON(n_layers > EDAC_MAX_LAYERS || n_layers == 0)) if (WARN_ON(n_layers > EDAC_MAX_LAYERS || n_layers == 0))
...@@ -482,19 +480,10 @@ struct mem_ctl_info *edac_mc_alloc(unsigned int mc_num, ...@@ -482,19 +480,10 @@ struct mem_ctl_info *edac_mc_alloc(unsigned int mc_num,
* stringent as what the compiler would provide if we could simply * stringent as what the compiler would provide if we could simply
* hardcode everything into a single struct. * hardcode everything into a single struct.
*/ */
mci = edac_align_ptr(&ptr, sizeof(*mci), 1); mci = edac_align_ptr(&ptr, sizeof(*mci), 1);
layer = edac_align_ptr(&ptr, sizeof(*layer), n_layers); layer = edac_align_ptr(&ptr, sizeof(*layer), n_layers);
for (i = 0; i < n_layers; i++) { pvt = edac_align_ptr(&ptr, sz_pvt, 1);
count *= layers[i].size; size = ((unsigned long)pvt) + sz_pvt;
edac_dbg(4, "errcount layer %d size %d\n", i, count);
ce_per_layer[i] = edac_align_ptr(&ptr, sizeof(u32), count);
ue_per_layer[i] = edac_align_ptr(&ptr, sizeof(u32), count);
tot_errcount += 2 * count;
}
edac_dbg(4, "allocating %d error counters\n", tot_errcount);
pvt = edac_align_ptr(&ptr, sz_pvt, 1);
size = ((unsigned long)pvt) + sz_pvt;
edac_dbg(1, "allocating %u bytes for mci data (%d %s, %d csrows/channels)\n", edac_dbg(1, "allocating %u bytes for mci data (%d %s, %d csrows/channels)\n",
size, size,
...@@ -513,10 +502,6 @@ struct mem_ctl_info *edac_mc_alloc(unsigned int mc_num, ...@@ -513,10 +502,6 @@ struct mem_ctl_info *edac_mc_alloc(unsigned int mc_num,
* rather than an imaginary chunk of memory located at address 0. * rather than an imaginary chunk of memory located at address 0.
*/ */
layer = (struct edac_mc_layer *)(((char *)mci) + ((unsigned long)layer)); layer = (struct edac_mc_layer *)(((char *)mci) + ((unsigned long)layer));
for (i = 0; i < n_layers; i++) {
mci->ce_per_layer[i] = (u32 *)((char *)mci + ((unsigned long)ce_per_layer[i]));
mci->ue_per_layer[i] = (u32 *)((char *)mci + ((unsigned long)ue_per_layer[i]));
}
pvt = sz_pvt ? (((char *)mci) + ((unsigned long)pvt)) : NULL; pvt = sz_pvt ? (((char *)mci) + ((unsigned long)pvt)) : NULL;
/* setup index and various internal pointers */ /* setup index and various internal pointers */
...@@ -949,48 +934,28 @@ static void edac_inc_ce_error(struct edac_raw_error_desc *e) ...@@ -949,48 +934,28 @@ static void edac_inc_ce_error(struct edac_raw_error_desc *e)
{ {
int pos[EDAC_MAX_LAYERS] = { e->top_layer, e->mid_layer, e->low_layer }; int pos[EDAC_MAX_LAYERS] = { e->top_layer, e->mid_layer, e->low_layer };
struct mem_ctl_info *mci = error_desc_to_mci(e); struct mem_ctl_info *mci = error_desc_to_mci(e);
int i, index = 0; struct dimm_info *dimm = edac_get_dimm(mci, pos[0], pos[1], pos[2]);
mci->ce_mc += e->error_count; mci->ce_mc += e->error_count;
if (pos[0] < 0) { if (dimm)
dimm->ce_count += e->error_count;
else
mci->ce_noinfo_count += e->error_count; mci->ce_noinfo_count += e->error_count;
return;
}
for (i = 0; i < mci->n_layers; i++) {
if (pos[i] < 0)
break;
index += pos[i];
mci->ce_per_layer[i][index] += e->error_count;
if (i < mci->n_layers - 1)
index *= mci->layers[i + 1].size;
}
} }
static void edac_inc_ue_error(struct edac_raw_error_desc *e) static void edac_inc_ue_error(struct edac_raw_error_desc *e)
{ {
int pos[EDAC_MAX_LAYERS] = { e->top_layer, e->mid_layer, e->low_layer }; int pos[EDAC_MAX_LAYERS] = { e->top_layer, e->mid_layer, e->low_layer };
struct mem_ctl_info *mci = error_desc_to_mci(e); struct mem_ctl_info *mci = error_desc_to_mci(e);
int i, index = 0; struct dimm_info *dimm = edac_get_dimm(mci, pos[0], pos[1], pos[2]);
mci->ue_mc += e->error_count; mci->ue_mc += e->error_count;
if (pos[0] < 0) { if (dimm)
dimm->ue_count += e->error_count;
else
mci->ue_noinfo_count += e->error_count; mci->ue_noinfo_count += e->error_count;
return;
}
for (i = 0; i < mci->n_layers; i++) {
if (pos[i] < 0)
break;
index += pos[i];
mci->ue_per_layer[i][index] += e->error_count;
if (i < mci->n_layers - 1)
index *= mci->layers[i + 1].size;
}
} }
static void edac_ce_error(struct edac_raw_error_desc *e) static void edac_ce_error(struct edac_raw_error_desc *e)
...@@ -1143,8 +1108,8 @@ void edac_mc_handle_error(const enum hw_event_mc_err_type type, ...@@ -1143,8 +1108,8 @@ void edac_mc_handle_error(const enum hw_event_mc_err_type type,
/* /*
* Check if the event report is consistent and if the memory location is * Check if the event report is consistent and if the memory location is
* known. If it is, the DIMM(s) label info will be filled and the * known. If it is, the DIMM(s) label info will be filled and the DIMM's
* per-layer error counters will be incremented. * error counters will be incremented.
*/ */
for (i = 0; i < mci->n_layers; i++) { for (i = 0; i < mci->n_layers; i++) {
if (pos[i] >= (int)mci->layers[i].size) { if (pos[i] >= (int)mci->layers[i].size) {
......
...@@ -551,10 +551,8 @@ static ssize_t dimmdev_ce_count_show(struct device *dev, ...@@ -551,10 +551,8 @@ static ssize_t dimmdev_ce_count_show(struct device *dev,
char *data) char *data)
{ {
struct dimm_info *dimm = to_dimm(dev); struct dimm_info *dimm = to_dimm(dev);
u32 count;
count = dimm->mci->ce_per_layer[dimm->mci->n_layers-1][dimm->idx]; return sprintf(data, "%u\n", dimm->ce_count);
return sprintf(data, "%u\n", count);
} }
static ssize_t dimmdev_ue_count_show(struct device *dev, static ssize_t dimmdev_ue_count_show(struct device *dev,
...@@ -562,10 +560,8 @@ static ssize_t dimmdev_ue_count_show(struct device *dev, ...@@ -562,10 +560,8 @@ static ssize_t dimmdev_ue_count_show(struct device *dev,
char *data) char *data)
{ {
struct dimm_info *dimm = to_dimm(dev); struct dimm_info *dimm = to_dimm(dev);
u32 count;
count = dimm->mci->ue_per_layer[dimm->mci->n_layers-1][dimm->idx]; return sprintf(data, "%u\n", dimm->ue_count);
return sprintf(data, "%u\n", count);
} }
/* dimm/rank attribute files */ /* dimm/rank attribute files */
...@@ -661,7 +657,9 @@ static ssize_t mci_reset_counters_store(struct device *dev, ...@@ -661,7 +657,9 @@ static ssize_t mci_reset_counters_store(struct device *dev,
const char *data, size_t count) const char *data, size_t count)
{ {
struct mem_ctl_info *mci = to_mci(dev); struct mem_ctl_info *mci = to_mci(dev);
int cnt, row, chan, i; struct dimm_info *dimm;
int row, chan;
mci->ue_mc = 0; mci->ue_mc = 0;
mci->ce_mc = 0; mci->ce_mc = 0;
mci->ue_noinfo_count = 0; mci->ue_noinfo_count = 0;
...@@ -677,11 +675,9 @@ static ssize_t mci_reset_counters_store(struct device *dev, ...@@ -677,11 +675,9 @@ static ssize_t mci_reset_counters_store(struct device *dev,
ri->channels[chan]->ce_count = 0; ri->channels[chan]->ce_count = 0;
} }
cnt = 1; mci_for_each_dimm(mci, dimm) {
for (i = 0; i < mci->n_layers; i++) { dimm->ue_count = 0;
cnt *= mci->layers[i].size; dimm->ce_count = 0;
memset(mci->ce_per_layer[i], 0, cnt * sizeof(u32));
memset(mci->ue_per_layer[i], 0, cnt * sizeof(u32));
} }
mci->start_time = jiffies; mci->start_time = jiffies;
......
...@@ -383,6 +383,9 @@ struct dimm_info { ...@@ -383,6 +383,9 @@ struct dimm_info {
unsigned int csrow, cschannel; /* Points to the old API data */ unsigned int csrow, cschannel; /* Points to the old API data */
u16 smbios_handle; /* Handle for SMBIOS type 17 */ u16 smbios_handle; /* Handle for SMBIOS type 17 */
u32 ce_count;
u32 ue_count;
}; };
/** /**
...@@ -559,7 +562,6 @@ struct mem_ctl_info { ...@@ -559,7 +562,6 @@ struct mem_ctl_info {
*/ */
u32 ce_noinfo_count, ue_noinfo_count; u32 ce_noinfo_count, ue_noinfo_count;
u32 ue_mc, ce_mc; u32 ue_mc, ce_mc;
u32 *ce_per_layer[EDAC_MAX_LAYERS], *ue_per_layer[EDAC_MAX_LAYERS];
struct completion complete; struct completion complete;
......
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