|
@@ -542,6 +542,99 @@ static const char *aer_agent_string[] = {
|
|
|
"Transmitter ID"
|
|
|
};
|
|
|
|
|
|
+#define aer_stats_dev_attr(name, stats_array, strings_array, \
|
|
|
+ total_string, total_field) \
|
|
|
+ static ssize_t \
|
|
|
+ name##_show(struct device *dev, struct device_attribute *attr, \
|
|
|
+ char *buf) \
|
|
|
+{ \
|
|
|
+ unsigned int i; \
|
|
|
+ char *str = buf; \
|
|
|
+ struct pci_dev *pdev = to_pci_dev(dev); \
|
|
|
+ u64 *stats = pdev->aer_stats->stats_array; \
|
|
|
+ \
|
|
|
+ for (i = 0; i < ARRAY_SIZE(strings_array); i++) { \
|
|
|
+ if (strings_array[i]) \
|
|
|
+ str += sprintf(str, "%s %llu\n", \
|
|
|
+ strings_array[i], stats[i]); \
|
|
|
+ else if (stats[i]) \
|
|
|
+ str += sprintf(str, #stats_array "_bit[%d] %llu\n",\
|
|
|
+ i, stats[i]); \
|
|
|
+ } \
|
|
|
+ str += sprintf(str, "TOTAL_%s %llu\n", total_string, \
|
|
|
+ pdev->aer_stats->total_field); \
|
|
|
+ return str-buf; \
|
|
|
+} \
|
|
|
+static DEVICE_ATTR_RO(name)
|
|
|
+
|
|
|
+aer_stats_dev_attr(aer_dev_correctable, dev_cor_errs,
|
|
|
+ aer_correctable_error_string, "ERR_COR",
|
|
|
+ dev_total_cor_errs);
|
|
|
+aer_stats_dev_attr(aer_dev_fatal, dev_fatal_errs,
|
|
|
+ aer_uncorrectable_error_string, "ERR_FATAL",
|
|
|
+ dev_total_fatal_errs);
|
|
|
+aer_stats_dev_attr(aer_dev_nonfatal, dev_nonfatal_errs,
|
|
|
+ aer_uncorrectable_error_string, "ERR_NONFATAL",
|
|
|
+ dev_total_nonfatal_errs);
|
|
|
+
|
|
|
+static struct attribute *aer_stats_attrs[] __ro_after_init = {
|
|
|
+ &dev_attr_aer_dev_correctable.attr,
|
|
|
+ &dev_attr_aer_dev_fatal.attr,
|
|
|
+ &dev_attr_aer_dev_nonfatal.attr,
|
|
|
+ NULL
|
|
|
+};
|
|
|
+
|
|
|
+static umode_t aer_stats_attrs_are_visible(struct kobject *kobj,
|
|
|
+ struct attribute *a, int n)
|
|
|
+{
|
|
|
+ struct device *dev = kobj_to_dev(kobj);
|
|
|
+ struct pci_dev *pdev = to_pci_dev(dev);
|
|
|
+
|
|
|
+ if (!pdev->aer_stats)
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ return a->mode;
|
|
|
+}
|
|
|
+
|
|
|
+const struct attribute_group aer_stats_attr_group = {
|
|
|
+ .attrs = aer_stats_attrs,
|
|
|
+ .is_visible = aer_stats_attrs_are_visible,
|
|
|
+};
|
|
|
+
|
|
|
+static void pci_dev_aer_stats_incr(struct pci_dev *pdev,
|
|
|
+ struct aer_err_info *info)
|
|
|
+{
|
|
|
+ int status, i, max = -1;
|
|
|
+ u64 *counter = NULL;
|
|
|
+ struct aer_stats *aer_stats = pdev->aer_stats;
|
|
|
+
|
|
|
+ if (!aer_stats)
|
|
|
+ return;
|
|
|
+
|
|
|
+ switch (info->severity) {
|
|
|
+ case AER_CORRECTABLE:
|
|
|
+ aer_stats->dev_total_cor_errs++;
|
|
|
+ counter = &aer_stats->dev_cor_errs[0];
|
|
|
+ max = AER_MAX_TYPEOF_COR_ERRS;
|
|
|
+ break;
|
|
|
+ case AER_NONFATAL:
|
|
|
+ aer_stats->dev_total_nonfatal_errs++;
|
|
|
+ counter = &aer_stats->dev_nonfatal_errs[0];
|
|
|
+ max = AER_MAX_TYPEOF_UNCOR_ERRS;
|
|
|
+ break;
|
|
|
+ case AER_FATAL:
|
|
|
+ aer_stats->dev_total_fatal_errs++;
|
|
|
+ counter = &aer_stats->dev_fatal_errs[0];
|
|
|
+ max = AER_MAX_TYPEOF_UNCOR_ERRS;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ status = (info->status & ~info->mask);
|
|
|
+ for (i = 0; i < max; i++)
|
|
|
+ if (status & (1 << i))
|
|
|
+ counter[i]++;
|
|
|
+}
|
|
|
+
|
|
|
static void __print_tlp_header(struct pci_dev *dev,
|
|
|
struct aer_header_log_regs *t)
|
|
|
{
|
|
@@ -574,6 +667,7 @@ static void __aer_print_error(struct pci_dev *dev,
|
|
|
pci_err(dev, " [%2d] Unknown Error Bit%s\n",
|
|
|
i, info->first_error == i ? " (First)" : "");
|
|
|
}
|
|
|
+ pci_dev_aer_stats_incr(dev, info);
|
|
|
}
|
|
|
|
|
|
void aer_print_error(struct pci_dev *dev, struct aer_err_info *info)
|