cxl: Fix force unmapping mmaps of contexts allocated through the kernel api
authorIan Munsie <imunsie@au1.ibm.com>
Thu, 27 Aug 2015 09:50:19 +0000 (19:50 +1000)
committerMichael Ellerman <mpe@ellerman.id.au>
Sun, 30 Aug 2015 08:47:26 +0000 (18:47 +1000)
The cxl user api uses the address_space associated with the file when we
need to force unmap all cxl mmap regions (e.g. on eeh, driver detach,
etc). Currently, contexts allocated through the kernel api do not do
this and instead skip the mmap invalidation, potentially allowing them
to poke at the hardware after such an event, which may cause all sorts
of trouble.

This patch allocates an address_space for cxl contexts allocated through
the kernel api so that the same invalidate path will for these contexts
as well. We don't use the anonymous inode's address_space, as doing so
could invalidate any mmaps of completely unrelated drivers using
anonymous file descriptors.

This patch also introduces a kernelapi flag, so we know when freeing the
context if the address_space was allocated by us and needs to be freed.

Signed-off-by: Ian Munsie <imunsie@au1.ibm.com>
Signed-off-by: Michael Ellerman <mpe@ellerman.id.au>
drivers/misc/cxl/api.c
drivers/misc/cxl/context.c
drivers/misc/cxl/cxl.h

index 005adc7d33a29c704a77da79b59b7fee36f8ac42..8af12c884b04eeb870d21ae2bb4bf15e0cfb76c4 100644 (file)
 #include <linux/anon_inodes.h>
 #include <linux/file.h>
 #include <misc/cxl.h>
+#include <linux/fs.h>
 
 #include "cxl.h"
 
 struct cxl_context *cxl_dev_context_init(struct pci_dev *dev)
 {
+       struct address_space *mapping;
        struct cxl_afu *afu;
        struct cxl_context  *ctx;
        int rc;
@@ -30,14 +32,32 @@ struct cxl_context *cxl_dev_context_init(struct pci_dev *dev)
                goto err_dev;
        }
 
+       ctx->kernelapi = true;
+
+       /*
+        * Make our own address space since we won't have one from the
+        * filesystem like the user api has, and even if we do associate a file
+        * with this context we don't want to use the global anonymous inode's
+        * address space as that can invalidate unrelated users:
+        */
+       mapping = kmalloc(sizeof(struct address_space), GFP_KERNEL);
+       if (!mapping) {
+               rc = -ENOMEM;
+               goto err_ctx;
+       }
+       address_space_init_once(mapping);
+
        /* Make it a slave context.  We can promote it later? */
-       rc = cxl_context_init(ctx, afu, false, NULL);
+       rc = cxl_context_init(ctx, afu, false, mapping);
        if (rc)
-               goto err_ctx;
+               goto err_mapping;
+
        cxl_assign_psn_space(ctx);
 
        return ctx;
 
+err_mapping:
+       kfree(mapping);
 err_ctx:
        kfree(ctx);
 err_dev:
@@ -260,9 +280,16 @@ struct file *cxl_get_fd(struct cxl_context *ctx, struct file_operations *fops,
 
        file = anon_inode_getfile("cxl", fops, ctx, flags);
        if (IS_ERR(file))
-               put_unused_fd(fdtmp);
+               goto err_fd;
+
+       file->f_mapping = ctx->mapping;
+
        *fd = fdtmp;
        return file;
+
+err_fd:
+       put_unused_fd(fdtmp);
+       return NULL;
 }
 EXPORT_SYMBOL_GPL(cxl_get_fd);
 
index 941fda04aa9a3092d1ddea19164348aeddbf7cb0..e762f85ee233a4b510390aa0ce4a5a79266b84c3 100644 (file)
@@ -272,6 +272,8 @@ static void reclaim_ctx(struct rcu_head *rcu)
        if (ctx->ff_page)
                __free_page(ctx->ff_page);
        ctx->sstp = NULL;
+       if (ctx->kernelapi)
+               kfree(ctx->mapping);
 
        kfree(ctx);
 }
index e7af256f60c54a97736ec7907a60171cab2eb90d..d6566c60bcb092571f27b43d1d67c93065459d7e 100644 (file)
@@ -420,6 +420,7 @@ struct cxl_context {
        struct mutex mapping_lock;
        struct page *ff_page;
        bool mmio_err_ff;
+       bool kernelapi;
 
        spinlock_t sste_lock; /* Protects segment table entries */
        struct cxl_sste *sstp;