pstore: Remove filesystem records when backend is unregistered
authorKees Cook <keescook@chromium.org>
Tue, 5 May 2020 02:46:53 +0000 (19:46 -0700)
committerKees Cook <keescook@chromium.org>
Sat, 30 May 2020 17:34:02 +0000 (10:34 -0700)
If a backend was unloaded without having first removed all its
associated records in pstorefs, subsequent removals would crash while
attempting to call into the now missing backend. Add automatic removal
from the tree in pstore_unregister(), so that no references to the
backend remain.

Reported-by: Luis Henriques <lhenriques@suse.com>
Link: https://lore.kernel.org/lkml/87o8yrmv69.fsf@suse.com
Link: https://lore.kernel.org/lkml/20200506152114.50375-11-keescook@chromium.org/
Signed-off-by: Kees Cook <keescook@chromium.org>
fs/pstore/inode.c
fs/pstore/internal.h
fs/pstore/platform.c

index 0e76e12fa6d13b19529f3d0232f037cc1fdb4eff..c331efe8de953732b22a385473357ab9ea9f2123 100644 (file)
@@ -36,6 +36,7 @@ static struct super_block *pstore_sb;
 
 struct pstore_private {
        struct list_head list;
+       struct dentry *dentry;
        struct pstore_record *record;
        size_t total_size;
 };
@@ -191,6 +192,7 @@ static int pstore_unlink(struct inode *dir, struct dentry *dentry)
                list_del_init(&p->list);
        else
                rc = -ENOENT;
+       p->dentry = NULL;
        mutex_unlock(&records_list_lock);
        if (rc)
                return rc;
@@ -306,6 +308,35 @@ static struct dentry *psinfo_lock_root(void)
        return root;
 }
 
+int pstore_put_backend_records(struct pstore_info *psi)
+{
+       struct pstore_private *pos, *tmp;
+       struct dentry *root;
+       int rc = 0;
+
+       root = psinfo_lock_root();
+       if (!root)
+               return 0;
+
+       mutex_lock(&records_list_lock);
+       list_for_each_entry_safe(pos, tmp, &records_list, list) {
+               if (pos->record->psi == psi) {
+                       list_del_init(&pos->list);
+                       rc = simple_unlink(d_inode(root), pos->dentry);
+                       if (WARN_ON(rc))
+                               break;
+                       d_drop(pos->dentry);
+                       dput(pos->dentry);
+                       pos->dentry = NULL;
+               }
+       }
+       mutex_unlock(&records_list_lock);
+
+       inode_unlock(d_inode(root));
+
+       return rc;
+}
+
 /*
  * Make a regular file in the root directory of our file system.
  * Load it up with "size" bytes of data from "buf".
@@ -352,6 +383,7 @@ int pstore_mkfile(struct dentry *root, struct pstore_record *record)
        if (!dentry)
                goto fail_private;
 
+       private->dentry = dentry;
        private->record = record;
        inode->i_size = private->total_size = size;
        inode->i_private = private;
index fe5f7ef7323f852fb0433aa61212389f696000f1..8efd72d93b10abca7fe26e42bb5243466a09c978 100644 (file)
@@ -31,6 +31,7 @@ extern void   pstore_set_kmsg_bytes(int);
 extern void    pstore_get_records(int);
 extern void    pstore_get_backend_records(struct pstore_info *psi,
                                           struct dentry *root, int quiet);
+extern int     pstore_put_backend_records(struct pstore_info *psi);
 extern int     pstore_mkfile(struct dentry *root,
                              struct pstore_record *record);
 extern void    pstore_record_init(struct pstore_record *record,
index 327ee70e881dcfe29cf6a12490a0875614aacbf5..398785ab059ff60e37b51607bb6a6e8e17e935e3 100644 (file)
@@ -657,6 +657,9 @@ void pstore_unregister(struct pstore_info *psi)
        del_timer_sync(&pstore_timer);
        flush_work(&pstore_work);
 
+       /* Remove all backend records from filesystem tree. */
+       pstore_put_backend_records(psi);
+
        free_buf_for_compression();
 
        psinfo = NULL;