This patchset avoids losing a critical message like panic in NVRAM.

Change v2 -> v3
 - Merged previous 1/3 and 2/3 patches.
 - Dropped previous a 3/3 patch.
 - Remove a kernel parameter ,efi_pstore_log_num.
 - Check if there is enough space to log with QueryVariableInfo().
 - Pass oopscount to arguments of write/erase/read callbacks
   to handle multiple events.

Change v1 -> v2

1/3
  - Freshly created to avoid overwriting entries.

2/3
  - Freshly created to handle multiple logs.
  - Add an additional change passing ctime to arguments of erase_callback.

3/3
  - This is based on previous 2/2 patch
  - Add comments to kernel/printk.h in preparation for future change 
    without considering this patch.
  - Remove infinite loop to avoid potential hang up.
  - Add CFLAGS, -Wswitch-enum and remove default case from switch sentence 
    in preparation for future change without considering this patch.
  - Change a return value to -EEXIST when an erasable entry is not found. 
  - Remove KMSG_DUMP_UNDEF from kmsg_dump_reason because no one uses it.


[Problem]
    Currently, efi_pstore driver simply overwrites existing panic messages in 
NVRAM.
    So, in the following scenario, we will lose 1st panic messages.

    1. kernel panics.
    2. efi_pstore is kicked and write panic messages to NVRAM.
    3. system reboots.
    4. kernel panics again before a user checks the 1st panic messages in NVRAM.

[Solution]
   To avoid losing a critical message, this patch takes an approach holding 
multiple
   logs.
   Also, to avoid handling out of space situation, it checks if there are 
enough spaces
   to write logs with QueryVariableInfo().


[Patch Descriptions]

  This patch makes efi_pstore hold multiple logs.
  Once a critical message is logged, users can see it via /dev/pstore 
  without being influenced by subsequent events.

  Specific changes are as follows.

  - Check if there are enough spaces to write logs with QueryVariableInfo().
  - Remove a logic erasing existing entries from write callback.
  - Add the logic above to erase callback.
  - Pass oopscount to arguments of write/erase/read callbacks to handle 
multiple events.
  - Pass ctime to arguments of erase_callback to avoid invisible entries via 
/dev/pstore.

 Signed-off-by: Seiji Aguchi <seiji.agu...@hds.com>
 Suggested-by: Matthew Garrett <m...@redhat.com>

---
 drivers/acpi/apei/erst.c   |   16 ++++----
 drivers/firmware/efivars.c |   93 ++++++++++++++++++++++++++++----------------
 fs/pstore/inode.c          |    7 ++-
 fs/pstore/internal.h       |    2 +-
 fs/pstore/platform.c       |   12 +++---
 fs/pstore/ram.c            |   11 ++---
 include/linux/efi.h        |    1 +
 include/linux/pstore.h     |    6 ++-
 8 files changed, 89 insertions(+), 59 deletions(-)

diff --git a/drivers/acpi/apei/erst.c b/drivers/acpi/apei/erst.c
index e4d9d24..833a9f4 100644
--- a/drivers/acpi/apei/erst.c
+++ b/drivers/acpi/apei/erst.c
@@ -931,14 +931,14 @@ static int erst_check_table(struct acpi_table_erst 
*erst_tab)
 
 static int erst_open_pstore(struct pstore_info *psi);
 static int erst_close_pstore(struct pstore_info *psi);
-static ssize_t erst_reader(u64 *id, enum pstore_type_id *type,
+static ssize_t erst_reader(u64 *id, enum pstore_type_id *type, u64 *count,
                           struct timespec *time, char **buf,
                           struct pstore_info *psi);
 static int erst_writer(enum pstore_type_id type, enum kmsg_dump_reason reason,
-                      u64 *id, unsigned int part,
+                      u64 *id, unsigned int part, u64 count,
                       size_t size, struct pstore_info *psi);
-static int erst_clearer(enum pstore_type_id type, u64 id,
-                       struct pstore_info *psi);
+static int erst_clearer(enum pstore_type_id type, u64 id, u64 count,
+                       struct timespec time, struct pstore_info *psi);
 
 static struct pstore_info erst_info = {
        .owner          = THIS_MODULE,
@@ -987,7 +987,7 @@ static int erst_close_pstore(struct pstore_info *psi)
        return 0;
 }
 
-static ssize_t erst_reader(u64 *id, enum pstore_type_id *type,
+static ssize_t erst_reader(u64 *id, enum pstore_type_id *type, u64 *count,
                           struct timespec *time, char **buf,
                           struct pstore_info *psi)
 {
@@ -1055,7 +1055,7 @@ out:
 }
 
 static int erst_writer(enum pstore_type_id type, enum kmsg_dump_reason reason,
-                      u64 *id, unsigned int part,
+                      u64 *id, unsigned int part, u64 count,
                       size_t size, struct pstore_info *psi)
 {
        struct cper_pstore_record *rcd = (struct cper_pstore_record *)
@@ -1101,8 +1101,8 @@ static int erst_writer(enum pstore_type_id type, enum 
kmsg_dump_reason reason,
        return ret;
 }
 
-static int erst_clearer(enum pstore_type_id type, u64 id,
-                       struct pstore_info *psi)
+static int erst_clearer(enum pstore_type_id type, u64 id, u64 count,
+                       struct timespec time, struct pstore_info *psi)
 {
        return erst_clear(id);
 }
diff --git a/drivers/firmware/efivars.c b/drivers/firmware/efivars.c
index 47408e8..9966fe3 100644
--- a/drivers/firmware/efivars.c
+++ b/drivers/firmware/efivars.c
@@ -647,7 +647,7 @@ static int efi_pstore_close(struct pstore_info *psi)
 }
 
 static ssize_t efi_pstore_read(u64 *id, enum pstore_type_id *type,
-                              struct timespec *timespec,
+                              u64 *count, struct timespec *timespec,
                               char **buf, struct pstore_info *psi)
 {
        efi_guid_t vendor = LINUX_EFI_CRASH_GUID;
@@ -655,6 +655,7 @@ static ssize_t efi_pstore_read(u64 *id, enum pstore_type_id 
*type,
        char name[DUMP_NAME_LEN];
        int i;
        unsigned int part, size;
+       u64 cnt;
        unsigned long time;
 
        while (&efivars->walk_entry->list != &efivars->list) {
@@ -663,8 +664,10 @@ static ssize_t efi_pstore_read(u64 *id, enum 
pstore_type_id *type,
                        for (i = 0; i < DUMP_NAME_LEN; i++) {
                                name[i] = 
efivars->walk_entry->var.VariableName[i];
                        }
-                       if (sscanf(name, "dump-type%u-%u-%lu", type, &part, 
&time) == 3) {
+                       if (sscanf(name, "dump-type%u-%u-%llu-%lu",
+                           type, &part, &cnt, &time) == 4) {
                                *id = part;
+                               *count = cnt;
                                timespec->tv_sec = time;
                                timespec->tv_nsec = 0;
                                get_var_data_locked(efivars, 
&efivars->walk_entry->var);
@@ -687,18 +690,62 @@ static ssize_t efi_pstore_read(u64 *id, enum 
pstore_type_id *type,
 
 static int efi_pstore_write(enum pstore_type_id type,
                enum kmsg_dump_reason reason, u64 *id,
-               unsigned int part, size_t size, struct pstore_info *psi)
+               unsigned int part, u64 count, size_t size,
+               struct pstore_info *psi)
 {
        char name[DUMP_NAME_LEN];
+       efi_char16_t efi_name[DUMP_NAME_LEN];
+       efi_guid_t vendor = LINUX_EFI_CRASH_GUID;
+       struct efivars *efivars = psi->data;
+       int i, ret = 0;
+       u64 storage_space, remaining_space, max_variable_size;
+       efi_status_t status = EFI_NOT_FOUND;
+
+       spin_lock(&efivars->lock);
+
+       status = efivars->ops->query_variable_info(PSTORE_EFI_ATTRIBUTES,
+                                                  &storage_space,
+                                                  &remaining_space,
+                                                  &max_variable_size);
+       if (status || remaining_space < max_variable_size) {
+               spin_unlock(&efivars->lock);
+               *id = part;
+               return -EEXIST;
+       }
+
+       sprintf(name, "dump-type%u-%u-%llu-%lu", type, part, count,
+               get_seconds());
+
+       for (i = 0; i < DUMP_NAME_LEN; i++)
+               efi_name[i] = name[i];
+
+       efivars->ops->set_variable(efi_name, &vendor, PSTORE_EFI_ATTRIBUTES,
+                                  size, psi->buf);
+
+       spin_unlock(&efivars->lock);
+
+       if (size)
+               ret = efivar_create_sysfs_entry(efivars,
+                                         utf16_strsize(efi_name,
+                                                       DUMP_NAME_LEN * 2),
+                                         efi_name, &vendor);
+
+       *id = part;
+       return ret;
+};
+
+static int efi_pstore_erase(enum pstore_type_id type, u64 id, u64 count,
+                           struct timespec time, struct pstore_info *psi)
+{
        char stub_name[DUMP_NAME_LEN];
        efi_char16_t efi_name[DUMP_NAME_LEN];
        efi_guid_t vendor = LINUX_EFI_CRASH_GUID;
        struct efivars *efivars = psi->data;
        struct efivar_entry *entry, *found = NULL;
-       int i, ret = 0;
+       int i;
 
-       sprintf(stub_name, "dump-type%u-%u-", type, part);
-       sprintf(name, "%s%lu", stub_name, get_seconds());
+       sprintf(stub_name, "dump-type%u-%llu-%llu-%lu", type, id, count,
+               time.tv_sec);
 
        spin_lock(&efivars->lock);
 
@@ -717,9 +764,6 @@ static int efi_pstore_write(enum pstore_type_id type,
                if (utf16_strncmp(entry->var.VariableName, efi_name,
                                  utf16_strlen(efi_name)))
                        continue;
-               /* Needs to be a prefix */
-               if (entry->var.VariableName[utf16_strlen(efi_name)] == 0)
-                       continue;
 
                /* found */
                found = entry;
@@ -732,32 +776,11 @@ static int efi_pstore_write(enum pstore_type_id type,
        if (found)
                list_del(&found->list);
 
-       for (i = 0; i < DUMP_NAME_LEN; i++)
-               efi_name[i] = name[i];
-
-       efivars->ops->set_variable(efi_name, &vendor, PSTORE_EFI_ATTRIBUTES,
-                                  size, psi->buf);
-
        spin_unlock(&efivars->lock);
 
        if (found)
                efivar_unregister(found);
 
-       if (size)
-               ret = efivar_create_sysfs_entry(efivars,
-                                         utf16_strsize(efi_name,
-                                                       DUMP_NAME_LEN * 2),
-                                         efi_name, &vendor);
-
-       *id = part;
-       return ret;
-};
-
-static int efi_pstore_erase(enum pstore_type_id type, u64 id,
-                           struct pstore_info *psi)
-{
-       efi_pstore_write(type, 0, &id, (unsigned int)id, 0, psi);
-
        return 0;
 }
 #else
@@ -771,7 +794,7 @@ static int efi_pstore_close(struct pstore_info *psi)
        return 0;
 }
 
-static ssize_t efi_pstore_read(u64 *id, enum pstore_type_id *type,
+static ssize_t efi_pstore_read(u64 *id, enum pstore_type_id *type, u64 *count,
                               struct timespec *timespec,
                               char **buf, struct pstore_info *psi)
 {
@@ -780,13 +803,14 @@ static ssize_t efi_pstore_read(u64 *id, enum 
pstore_type_id *type,
 
 static int efi_pstore_write(enum pstore_type_id type,
                enum kmsg_dump_reason reason, u64 *id,
-               unsigned int part, size_t size, struct pstore_info *psi)
+               unsigned int part, u64 count, size_t size,
+               struct pstore_info *psi)
 {
        return 0;
 }
 
-static int efi_pstore_erase(enum pstore_type_id type, u64 id,
-                           struct pstore_info *psi)
+static int efi_pstore_erase(enum pstore_type_id type, u64 id, u64 count,
+                           struct timespec time, struct pstore_info *psi)
 {
        return 0;
 }
@@ -1226,6 +1250,7 @@ efivars_init(void)
        ops.get_variable = efi.get_variable;
        ops.set_variable = efi.set_variable;
        ops.get_next_variable = efi.get_next_variable;
+       ops.query_variable_info = efi.query_variable_info;
        error = register_efivars(&__efivars, &ops, efi_kobj);
        if (error)
                goto err_put;
diff --git a/fs/pstore/inode.c b/fs/pstore/inode.c
index 11a2aa2..ca1b6c9 100644
--- a/fs/pstore/inode.c
+++ b/fs/pstore/inode.c
@@ -48,6 +48,7 @@ struct pstore_private {
        struct pstore_info *psi;
        enum pstore_type_id type;
        u64     id;
+       u64     count;
        ssize_t size;
        char    data[];
 };
@@ -75,7 +76,8 @@ static int pstore_unlink(struct inode *dir, struct dentry 
*dentry)
        struct pstore_private *p = dentry->d_inode->i_private;
 
        if (p->psi->erase)
-               p->psi->erase(p->type, p->id, p->psi);
+               p->psi->erase(p->type, p->id, p->count,
+                             dentry->d_inode->i_ctime, p->psi);
 
        return simple_unlink(dir, dentry);
 }
@@ -170,7 +172,7 @@ int pstore_is_mounted(void)
  * Load it up with "size" bytes of data from "buf".
  * Set the mtime & ctime to the date that this record was originally stored.
  */
-int pstore_mkfile(enum pstore_type_id type, char *psname, u64 id,
+int pstore_mkfile(enum pstore_type_id type, char *psname, u64 id, u64 count,
                  char *data, size_t size, struct timespec time,
                  struct pstore_info *psi)
 {
@@ -206,6 +208,7 @@ int pstore_mkfile(enum pstore_type_id type, char *psname, 
u64 id,
                goto fail_alloc;
        private->type = type;
        private->id = id;
+       private->count = count;
        private->psi = psi;
 
        switch (type) {
diff --git a/fs/pstore/internal.h b/fs/pstore/internal.h
index 3bde461..d93e20e 100644
--- a/fs/pstore/internal.h
+++ b/fs/pstore/internal.h
@@ -1,6 +1,6 @@
 extern void    pstore_set_kmsg_bytes(int);
 extern void    pstore_get_records(int);
 extern int     pstore_mkfile(enum pstore_type_id, char *psname, u64 id,
-                             char *data, size_t size,
+                             u64 count, char *data, size_t size,
                              struct timespec time, struct pstore_info *psi);
 extern int     pstore_is_mounted(void);
diff --git a/fs/pstore/platform.c b/fs/pstore/platform.c
index 03ce7a9..371a184 100644
--- a/fs/pstore/platform.c
+++ b/fs/pstore/platform.c
@@ -66,7 +66,7 @@ void pstore_set_kmsg_bytes(int bytes)
 }
 
 /* Tag each group of saved records with a sequence number */
-static int     oopscount;
+unsigned int   oopscount;
 
 static const char *get_reason_str(enum kmsg_dump_reason reason)
 {
@@ -128,7 +128,7 @@ static void pstore_dump(struct kmsg_dumper *dumper,
                        break;
 
                ret = psinfo->write(PSTORE_TYPE_DMESG, reason, &id, part,
-                                   hsize + len, psinfo);
+                                   oopscount, hsize + len, psinfo);
                if (ret == 0 && reason == KMSG_DUMP_OOPS && pstore_is_mounted())
                        pstore_new_entry = 1;
 
@@ -202,7 +202,7 @@ void pstore_get_records(int quiet)
        struct pstore_info *psi = psinfo;
        char                    *buf = NULL;
        ssize_t                 size;
-       u64                     id;
+       u64                     id, count;
        enum pstore_type_id     type;
        struct timespec         time;
        int                     failed = 0, rc;
@@ -214,9 +214,9 @@ void pstore_get_records(int quiet)
        if (psi->open && psi->open(psi))
                goto out;
 
-       while ((size = psi->read(&id, &type, &time, &buf, psi)) > 0) {
-               rc = pstore_mkfile(type, psi->name, id, buf, (size_t)size,
-                                 time, psi);
+       while ((size = psi->read(&id, &type, &count, &time, &buf, psi)) > 0) {
+               rc = pstore_mkfile(type, psi->name, id, count, buf,
+                                  (size_t)size, time, psi);
                kfree(buf);
                buf = NULL;
                if (rc && (rc != -EEXIST || !quiet))
diff --git a/fs/pstore/ram.c b/fs/pstore/ram.c
index 453030f..2bdd4f2 100644
--- a/fs/pstore/ram.c
+++ b/fs/pstore/ram.c
@@ -86,9 +86,8 @@ static int ramoops_pstore_open(struct pstore_info *psi)
 }
 
 static ssize_t ramoops_pstore_read(u64 *id, enum pstore_type_id *type,
-                                  struct timespec *time,
-                                  char **buf,
-                                  struct pstore_info *psi)
+                                  u64 *count, struct timespec *time,
+                                  char **buf, struct pstore_info *psi)
 {
        ssize_t size;
        struct ramoops_context *cxt = psi->data;
@@ -137,7 +136,7 @@ static size_t ramoops_write_kmsg_hdr(struct 
persistent_ram_zone *prz)
 static int ramoops_pstore_write(enum pstore_type_id type,
                                enum kmsg_dump_reason reason,
                                u64 *id,
-                               unsigned int part,
+                               unsigned int part, u64 count,
                                size_t size, struct pstore_info *psi)
 {
        struct ramoops_context *cxt = psi->data;
@@ -177,8 +176,8 @@ static int ramoops_pstore_write(enum pstore_type_id type,
        return 0;
 }
 
-static int ramoops_pstore_erase(enum pstore_type_id type, u64 id,
-                               struct pstore_info *psi)
+static int ramoops_pstore_erase(enum pstore_type_id type, u64 id, u64 count,
+                               timespec time, struct pstore_info *psi)
 {
        struct ramoops_context *cxt = psi->data;
 
diff --git a/include/linux/efi.h b/include/linux/efi.h
index ec45ccd..b19bef5 100644
--- a/include/linux/efi.h
+++ b/include/linux/efi.h
@@ -635,6 +635,7 @@ struct efivar_operations {
        efi_get_variable_t *get_variable;
        efi_get_next_variable_t *get_next_variable;
        efi_set_variable_t *set_variable;
+       efi_query_variable_info_t *query_variable_info;
 };
 
 struct efivars {
diff --git a/include/linux/pstore.h b/include/linux/pstore.h
index e1461e1..9fd1fc8 100644
--- a/include/linux/pstore.h
+++ b/include/linux/pstore.h
@@ -42,12 +42,14 @@ struct pstore_info {
        int             (*open)(struct pstore_info *psi);
        int             (*close)(struct pstore_info *psi);
        ssize_t         (*read)(u64 *id, enum pstore_type_id *type,
-                       struct timespec *time, char **buf,
+                       u64 *count, struct timespec *time, char **buf,
                        struct pstore_info *psi);
        int             (*write)(enum pstore_type_id type,
                        enum kmsg_dump_reason reason, u64 *id,
-                       unsigned int part, size_t size, struct pstore_info 
*psi);
+                       unsigned int part, u64 count, size_t size,
+                       struct pstore_info *psi);
        int             (*erase)(enum pstore_type_id type, u64 id,
+                       u64 count, struct timespec time,
                        struct pstore_info *psi);
        void            *data;
 };
-- 1.7.1
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to