ACPI: APEI: fix the wrong iteration of generic error status block
authorgengdongjiu <gengdongjiu@huawei.com>
Thu, 17 Aug 2017 12:07:18 +0000 (20:07 +0800)
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>
Thu, 24 Aug 2017 01:29:46 +0000 (03:29 +0200)
The revision 0x300 generic error data entry is different
from the old version, but currently iterating through the
GHES estatus blocks does not take into account this difference.
This will lead to failure to get the right data entry if GHES
has revision 0x300 error data entry.

Update the GHES estatus iteration macro to properly increment using
acpi_hest_get_next(), and correct the iteration termination condition
because the status block data length only includes error data
length.

Convert the CPER estatus checking and printing iteration logic
to use same macro.

Signed-off-by: Dongjiu Geng <gengdongjiu@huawei.com>
Tested-by: Tyler Baicar <tbaicar@codeaurora.org>
Reviewed-by: Borislav Petkov <bp@suse.de>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
drivers/acpi/apei/apei-internal.h
drivers/firmware/efi/cper.c
include/acpi/ghes.h

index 6e9f14c0a71bd6365b5aeac106d428a4b8161b8d..cb4126051f6220ba18bbd793e7b1684bc3fc448a 100644 (file)
@@ -120,11 +120,6 @@ int apei_exec_collect_resources(struct apei_exec_context *ctx,
 struct dentry;
 struct dentry *apei_get_debugfs_dir(void);
 
-#define apei_estatus_for_each_section(estatus, section)                        \
-       for (section = (struct acpi_hest_generic_data *)(estatus + 1);  \
-            (void *)section - (void *)estatus < estatus->data_length;  \
-            section = (void *)(section+1) + section->error_data_length)
-
 static inline u32 cper_estatus_len(struct acpi_hest_generic_status *estatus)
 {
        if (estatus->raw_data_length)
index 48a8f69da42aabacffe8efbbf0bae3e2d38669f3..bf3672a81e4948e87e494e4199cf6a16c1692793 100644 (file)
@@ -606,7 +606,6 @@ void cper_estatus_print(const char *pfx,
                        const struct acpi_hest_generic_status *estatus)
 {
        struct acpi_hest_generic_data *gdata;
-       unsigned int data_len;
        int sec_no = 0;
        char newpfx[64];
        __u16 severity;
@@ -617,14 +616,10 @@ void cper_estatus_print(const char *pfx,
                       "It has been corrected by h/w "
                       "and requires no further action");
        printk("%s""event severity: %s\n", pfx, cper_severity_str(severity));
-       data_len = estatus->data_length;
-       gdata = (struct acpi_hest_generic_data *)(estatus + 1);
        snprintf(newpfx, sizeof(newpfx), "%s%s", pfx, INDENT_SP);
 
-       while (data_len >= acpi_hest_get_size(gdata)) {
+       apei_estatus_for_each_section(estatus, gdata) {
                cper_estatus_print_section(newpfx, gdata, sec_no);
-               data_len -= acpi_hest_get_record_size(gdata);
-               gdata = acpi_hest_get_next(gdata);
                sec_no++;
        }
 }
@@ -653,15 +648,12 @@ int cper_estatus_check(const struct acpi_hest_generic_status *estatus)
        if (rc)
                return rc;
        data_len = estatus->data_length;
-       gdata = (struct acpi_hest_generic_data *)(estatus + 1);
 
-       while (data_len >= acpi_hest_get_size(gdata)) {
+       apei_estatus_for_each_section(estatus, gdata) {
                gedata_len = acpi_hest_get_error_length(gdata);
                if (gedata_len > data_len - acpi_hest_get_size(gdata))
                        return -EINVAL;
-
                data_len -= acpi_hest_get_record_size(gdata);
-               gdata = acpi_hest_get_next(gdata);
        }
        if (data_len)
                return -EINVAL;
index 9f26e01186aea6ccd37d1c7771de3978d9111bb5..9061c5c743b386cb05e065d1158a9fd1ca771dae 100644 (file)
@@ -113,6 +113,11 @@ static inline void *acpi_hest_get_next(struct acpi_hest_generic_data *gdata)
        return (void *)(gdata) + acpi_hest_get_record_size(gdata);
 }
 
+#define apei_estatus_for_each_section(estatus, section)                        \
+       for (section = (struct acpi_hest_generic_data *)(estatus + 1);  \
+            (void *)section - (void *)(estatus + 1) < estatus->data_length; \
+            section = acpi_hest_get_next(section))
+
 int ghes_notify_sea(void);
 
 #endif /* GHES_H */