u32 effects = 0;
 
        if (ns) {
-               if (ctrl->effects)
-                       effects = le32_to_cpu(ctrl->effects->iocs[opcode]);
+               if (ns->head->effects)
+                       effects = le32_to_cpu(ns->head->effects->iocs[opcode]);
                if (effects & ~(NVME_CMD_EFFECTS_CSUPP | NVME_CMD_EFFECTS_LBCC))
                        dev_warn(ctrl->device,
                                 "IO command:%02x has unhandled effects:%08x\n",
        return ret;
 }
 
-int nvme_get_log(struct nvme_ctrl *ctrl, u32 nsid, u8 log_page, u8 lsp,
+int nvme_get_log(struct nvme_ctrl *ctrl, u32 nsid, u8 log_page, u8 lsp, u8 csi,
                void *log, size_t size, u64 offset)
 {
        struct nvme_command c = { };
        c.get_log_page.numdu = cpu_to_le16(dwlen >> 16);
        c.get_log_page.lpol = cpu_to_le32(lower_32_bits(offset));
        c.get_log_page.lpou = cpu_to_le32(upper_32_bits(offset));
+       c.get_log_page.csi = csi;
 
        return nvme_submit_sync_cmd(ctrl->admin_q, &c, log, size);
 }
 
-static int nvme_get_effects_log(struct nvme_ctrl *ctrl)
+static struct nvme_cel *nvme_find_cel(struct nvme_ctrl *ctrl, u8 csi)
 {
+       struct nvme_cel *cel, *ret = NULL;
+
+       spin_lock(&ctrl->lock);
+       list_for_each_entry(cel, &ctrl->cels, entry) {
+               if (cel->csi == csi) {
+                       ret = cel;
+                       break;
+               }
+       }
+       spin_unlock(&ctrl->lock);
+
+       return ret;
+}
+
+static int nvme_get_effects_log(struct nvme_ctrl *ctrl, u8 csi,
+                               struct nvme_effects_log **log)
+{
+       struct nvme_cel *cel = nvme_find_cel(ctrl, csi);
        int ret;
 
-       if (!ctrl->effects)
-               ctrl->effects = kzalloc(sizeof(*ctrl->effects), GFP_KERNEL);
+       if (cel)
+               goto out;
 
-       if (!ctrl->effects)
-               return 0;
+       cel = kzalloc(sizeof(*cel), GFP_KERNEL);
+       if (!cel)
+               return -ENOMEM;
 
-       ret = nvme_get_log(ctrl, NVME_NSID_ALL, NVME_LOG_CMD_EFFECTS, 0,
-                       ctrl->effects, sizeof(*ctrl->effects), 0);
+       ret = nvme_get_log(ctrl, NVME_NSID_ALL, NVME_LOG_CMD_EFFECTS, 0, csi,
+                       &cel->log, sizeof(cel->log), 0);
        if (ret) {
-               kfree(ctrl->effects);
-               ctrl->effects = NULL;
+               kfree(cel);
+               return ret;
        }
-       return ret;
+
+       cel->csi = csi;
+
+       spin_lock(&ctrl->lock);
+       list_add_tail(&cel->entry, &ctrl->cels);
+       spin_unlock(&ctrl->lock);
+out:
+       *log = &cel->log;
+       return 0;
 }
 
 /*
        }
 
        if (id->lpa & NVME_CTRL_LPA_CMD_EFFECTS_LOG) {
-               ret = nvme_get_effects_log(ctrl);
+               ret = nvme_get_effects_log(ctrl, NVME_CSI_NVM, &ctrl->effects);
                if (ret < 0)
                        goto out_free;
        }
                goto out_cleanup_srcu;
        }
 
+       if (head->ids.csi) {
+               ret = nvme_get_effects_log(ctrl, head->ids.csi, &head->effects);
+               if (ret)
+                       goto out_cleanup_srcu;
+       } else
+               head->effects = ctrl->effects;
+
        ret = nvme_mpath_alloc_disk(ctrl, head);
        if (ret)
                goto out_cleanup_srcu;
         * raced with us in reading the log page, which could cause us to miss
         * updates.
         */
-       error = nvme_get_log(ctrl, NVME_NSID_ALL, NVME_LOG_CHANGED_NS, 0, log,
-                       log_size, 0);
+       error = nvme_get_log(ctrl, NVME_NSID_ALL, NVME_LOG_CHANGED_NS, 0,
+                       NVME_CSI_NVM, log, log_size, 0);
        if (error)
                dev_warn(ctrl->device,
                        "reading changed ns log failed: %d\n", error);
        if (!log)
                return;
 
-       if (nvme_get_log(ctrl, NVME_NSID_ALL, NVME_LOG_FW_SLOT, 0, log,
-                       sizeof(*log), 0))
+       if (nvme_get_log(ctrl, NVME_NSID_ALL, NVME_LOG_FW_SLOT, 0, NVME_CSI_NVM,
+                       log, sizeof(*log), 0))
                dev_warn(ctrl->device, "Get FW SLOT INFO log error\n");
        kfree(log);
 }
        struct nvme_ctrl *ctrl =
                container_of(dev, struct nvme_ctrl, ctrl_device);
        struct nvme_subsystem *subsys = ctrl->subsys;
+       struct nvme_cel *cel, *next;
 
        if (subsys && ctrl->instance != subsys->instance)
                ida_simple_remove(&nvme_instance_ida, ctrl->instance);
 
-       kfree(ctrl->effects);
+       list_for_each_entry_safe(cel, next, &ctrl->cels, entry) {
+               list_del(&cel->entry);
+               kfree(cel);
+       }
+
        nvme_mpath_uninit(ctrl);
        __free_page(ctrl->discard_page);
 
        spin_lock_init(&ctrl->lock);
        mutex_init(&ctrl->scan_lock);
        INIT_LIST_HEAD(&ctrl->namespaces);
+       INIT_LIST_HEAD(&ctrl->cels);
        init_rwsem(&ctrl->namespaces_rwsem);
        ctrl->dev = dev;
        ctrl->ops = ops;
 
 #endif
 };
 
+struct nvme_cel {
+       struct list_head        entry;
+       struct nvme_effects_log log;
+       u8                      csi;
+};
+
 struct nvme_ctrl {
        bool comp_seen;
        enum nvme_ctrl_state state;
        unsigned long quirks;
        struct nvme_id_power_state psd[32];
        struct nvme_effects_log *effects;
+       struct list_head cels;
        struct work_struct scan_work;
        struct work_struct async_event_work;
        struct delayed_work ka_work;
        struct kref             ref;
        bool                    shared;
        int                     instance;
+       struct nvme_effects_log *effects;
 #ifdef CONFIG_NVME_MULTIPATH
        struct gendisk          *disk;
        struct bio_list         requeue_list;
 int nvme_try_sched_reset(struct nvme_ctrl *ctrl);
 int nvme_delete_ctrl(struct nvme_ctrl *ctrl);
 
-int nvme_get_log(struct nvme_ctrl *ctrl, u32 nsid, u8 log_page, u8 lsp,
+int nvme_get_log(struct nvme_ctrl *ctrl, u32 nsid, u8 log_page, u8 lsp, u8 csi,
                void *log, size_t size, u64 offset);
 
 extern const struct attribute_group *nvme_ns_id_attr_groups[];