scsi: cxlflash: Introduce OCXL context state machine
authorUma Krishnan <ukrishn@linux.vnet.ibm.com>
Mon, 26 Mar 2018 16:35:00 +0000 (11:35 -0500)
committerMartin K. Petersen <martin.petersen@oracle.com>
Wed, 18 Apr 2018 23:32:50 +0000 (19:32 -0400)
In order to protect the OCXL hardware contexts from getting clobbered, a
simple state machine is added to indicate when a context is in open, close or
start state. The expected states are validated throughout the code to prevent
illegal operations on a context. A mutex is added to protect writes to the
context state field.

Signed-off-by: Uma Krishnan <ukrishn@linux.vnet.ibm.com>
Acked-by: Matthew R. Ochs <mrochs@linux.vnet.ibm.com>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
drivers/scsi/cxlflash/ocxl_hw.c
drivers/scsi/cxlflash/ocxl_hw.h

index f77f4d7f6a34f76b7c651ae22ccd21452736c962..89d3d89f22ed1302be22bc6954a28fb491fa0f95 100644 (file)
@@ -163,6 +163,16 @@ err1:
 static void __iomem *ocxlflash_psa_map(void *ctx_cookie)
 {
        struct ocxlflash_context *ctx = ctx_cookie;
+       struct device *dev = ctx->hw_afu->dev;
+
+       mutex_lock(&ctx->state_mutex);
+       if (ctx->state != STARTED) {
+               dev_err(dev, "%s: Context not started, state=%d\n", __func__,
+                       ctx->state);
+               mutex_unlock(&ctx->state_mutex);
+               return NULL;
+       }
+       mutex_unlock(&ctx->state_mutex);
 
        return ioremap(ctx->psn_phys, ctx->psn_size);
 }
@@ -343,6 +353,14 @@ static int start_context(struct ocxlflash_context *ctx)
        int rc = 0;
        u32 pid;
 
+       mutex_lock(&ctx->state_mutex);
+       if (ctx->state != OPENED) {
+               dev_err(dev, "%s: Context state invalid, state=%d\n",
+                       __func__, ctx->state);
+               rc = -EINVAL;
+               goto out;
+       }
+
        if (master) {
                ctx->psn_size = acfg->global_mmio_size;
                ctx->psn_phys = afu->gmmio_phys;
@@ -366,7 +384,10 @@ static int start_context(struct ocxlflash_context *ctx)
                        __func__, rc);
                goto out;
        }
+
+       ctx->state = STARTED;
 out:
+       mutex_unlock(&ctx->state_mutex);
        return rc;
 }
 
@@ -396,7 +417,15 @@ static int ocxlflash_stop_context(void *ctx_cookie)
        struct ocxl_afu_config *acfg = &afu->acfg;
        struct pci_dev *pdev = afu->pdev;
        struct device *dev = afu->dev;
-       int rc;
+       enum ocxlflash_ctx_state state;
+       int rc = 0;
+
+       mutex_lock(&ctx->state_mutex);
+       state = ctx->state;
+       ctx->state = CLOSED;
+       mutex_unlock(&ctx->state_mutex);
+       if (state != STARTED)
+               goto out;
 
        rc = ocxl_config_terminate_pasid(pdev, acfg->dvsec_afu_control_pos,
                                         ctx->pe);
@@ -474,7 +503,9 @@ static void *ocxlflash_dev_context_init(struct pci_dev *pdev, void *afu_cookie)
 
        spin_lock_init(&ctx->slock);
        init_waitqueue_head(&ctx->wq);
+       mutex_init(&ctx->state_mutex);
 
+       ctx->state = OPENED;
        ctx->pe = rc;
        ctx->master = false;
        ctx->mapping = NULL;
@@ -499,11 +530,23 @@ err1:
 static int ocxlflash_release_context(void *ctx_cookie)
 {
        struct ocxlflash_context *ctx = ctx_cookie;
+       struct device *dev;
        int rc = 0;
 
        if (!ctx)
                goto out;
 
+       dev = ctx->hw_afu->dev;
+       mutex_lock(&ctx->state_mutex);
+       if (ctx->state >= STARTED) {
+               dev_err(dev, "%s: Context in use, state=%d\n", __func__,
+                       ctx->state);
+               mutex_unlock(&ctx->state_mutex);
+               rc = -EBUSY;
+               goto out;
+       }
+       mutex_unlock(&ctx->state_mutex);
+
        idr_remove(&ctx->hw_afu->idr, ctx->pe);
        ocxlflash_release_mapping(ctx);
        kfree(ctx);
@@ -947,7 +990,7 @@ static unsigned int afu_poll(struct file *file, struct poll_table_struct *poll)
        spin_lock_irqsave(&ctx->slock, lock_flags);
        if (ctx_event_pending(ctx))
                mask |= POLLIN | POLLRDNORM;
-       else
+       else if (ctx->state == CLOSED)
                mask |= POLLERR;
        spin_unlock_irqrestore(&ctx->slock, lock_flags);
 
@@ -990,7 +1033,7 @@ static ssize_t afu_read(struct file *file, char __user *buf, size_t count,
        for (;;) {
                prepare_to_wait(&ctx->wq, &event_wait, TASK_INTERRUPTIBLE);
 
-               if (ctx_event_pending(ctx))
+               if (ctx_event_pending(ctx) || (ctx->state == CLOSED))
                        break;
 
                if (file->f_flags & O_NONBLOCK) {
@@ -1076,12 +1119,22 @@ static int ocxlflash_mmap_fault(struct vm_fault *vmf)
 {
        struct vm_area_struct *vma = vmf->vma;
        struct ocxlflash_context *ctx = vma->vm_file->private_data;
+       struct device *dev = ctx->hw_afu->dev;
        u64 mmio_area, offset;
 
        offset = vmf->pgoff << PAGE_SHIFT;
        if (offset >= ctx->psn_size)
                return VM_FAULT_SIGBUS;
 
+       mutex_lock(&ctx->state_mutex);
+       if (ctx->state != STARTED) {
+               dev_err(dev, "%s: Context not started, state=%d\n",
+                       __func__, ctx->state);
+               mutex_unlock(&ctx->state_mutex);
+               return VM_FAULT_SIGBUS;
+       }
+       mutex_unlock(&ctx->state_mutex);
+
        mmio_area = ctx->psn_phys;
        mmio_area += offset;
 
index acd2801359898daf7570852bea6c55361f2eb73e..1829e55c974a8036b4851cea8bc35e19e78a7724 100644 (file)
@@ -46,6 +46,12 @@ struct ocxl_hw_afu {
        bool is_present;                /* Function has AFUs defined */
 };
 
+enum ocxlflash_ctx_state {
+       CLOSED,
+       OPENED,
+       STARTED
+};
+
 struct ocxlflash_context {
        struct ocxl_hw_afu *hw_afu;     /* HW AFU back pointer */
        struct address_space *mapping;  /* Mapping for pseudo filesystem */
@@ -57,6 +63,8 @@ struct ocxlflash_context {
 
        spinlock_t slock;               /* Protects irq/fault/event updates */
        wait_queue_head_t wq;           /* Wait queue for poll and interrupts */
+       struct mutex state_mutex;       /* Mutex to update context state */
+       enum ocxlflash_ctx_state state; /* Context state */
 
        struct ocxlflash_irqs *irqs;    /* Pointer to array of structures */
        int num_irqs;                   /* Number of interrupts */