nvme: use page-aligned buffer for identify command
authorPatrick Wildt <patrick@blueri.se>
Wed, 16 Oct 2019 06:42:04 +0000 (08:42 +0200)
committerTom Rini <trini@konsulko.com>
Thu, 31 Oct 2019 11:22:53 +0000 (07:22 -0400)
Change the stack-allocated buffer for the identification command
to explicitly allocate page-aligned buffers.  Even though the spec
seems to allow having admin queue commands on non page-aligned
buffers, it seems to not be possible on my i.MX8MQ board with a
a Silicon Power P34A80.  Since all of the NVMe drivers I have seen
always do admin commands on a page-aligned buffer, which does work
on my system, it makes sense for us to do that as well.

Signed-off-by: Patrick Wildt <patrick@blueri.se>
Reviewed-by: Bin Meng <bmeng.cn@gmail.com>
drivers/nvme/nvme.c

index 53ff6e89aa27ad92893ccff6f4ca2e94e227c2a4..f915817aaa431cfbc4f0fad5457ef8674f20d546 100644 (file)
@@ -583,14 +583,19 @@ static int nvme_setup_io_queues(struct nvme_dev *dev)
 
 static int nvme_get_info_from_identify(struct nvme_dev *dev)
 {
-       ALLOC_CACHE_ALIGN_BUFFER(char, buf, sizeof(struct nvme_id_ctrl));
-       struct nvme_id_ctrl *ctrl = (struct nvme_id_ctrl *)buf;
+       struct nvme_id_ctrl *ctrl;
        int ret;
        int shift = NVME_CAP_MPSMIN(dev->cap) + 12;
 
+       ctrl = memalign(dev->page_size, sizeof(struct nvme_id_ctrl));
+       if (!ctrl)
+               return -ENOMEM;
+
        ret = nvme_identify(dev, 0, 1, (dma_addr_t)(long)ctrl);
-       if (ret)
+       if (ret) {
+               free(ctrl);
                return -EIO;
+       }
 
        dev->nn = le32_to_cpu(ctrl->nn);
        dev->vwc = ctrl->vwc;
@@ -621,6 +626,7 @@ static int nvme_get_info_from_identify(struct nvme_dev *dev)
                dev->max_transfer_shift = 20;
        }
 
+       free(ctrl);
        return 0;
 }
 
@@ -661,16 +667,21 @@ static int nvme_blk_probe(struct udevice *udev)
        struct blk_desc *desc = dev_get_uclass_platdata(udev);
        struct nvme_ns *ns = dev_get_priv(udev);
        u8 flbas;
-       ALLOC_CACHE_ALIGN_BUFFER(char, buf, sizeof(struct nvme_id_ns));
-       struct nvme_id_ns *id = (struct nvme_id_ns *)buf;
        struct pci_child_platdata *pplat;
+       struct nvme_id_ns *id;
+
+       id = memalign(ndev->page_size, sizeof(struct nvme_id_ns));
+       if (!id)
+               return -ENOMEM;
 
        memset(ns, 0, sizeof(*ns));
        ns->dev = ndev;
        /* extract the namespace id from the block device name */
        ns->ns_id = trailing_strtol(udev->name) + 1;
-       if (nvme_identify(ndev, ns->ns_id, 0, (dma_addr_t)(long)id))
+       if (nvme_identify(ndev, ns->ns_id, 0, (dma_addr_t)(long)id)) {
+               free(id);
                return -EIO;
+       }
 
        memcpy(&ns->eui64, &id->eui64, sizeof(id->eui64));
        flbas = id->flbas & NVME_NS_FLBAS_LBA_MASK;
@@ -689,6 +700,7 @@ static int nvme_blk_probe(struct udevice *udev)
        memcpy(desc->product, ndev->serial, sizeof(ndev->serial));
        memcpy(desc->revision, ndev->firmware_rev, sizeof(ndev->firmware_rev));
 
+       free(id);
        return 0;
 }