mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-12-22 18:44:44 +08:00
1ffc6f359f
The commit in the Fixes: tag has changed the logic of the code and now it
is likely that the probe will return an early success (0), even if not
completely executed.
This should lead to a crash or similar issue later on when the code
accesses to some never allocated resources.
Change the '!err' into a 'err' when checking if
'dma_set_mask_and_coherent()' has failed or not.
While at it, simplify the code and remove the "can't success code" related
to 32 DMA mask.
As stated in [1], 'dma_set_mask_and_coherent(DMA_BIT_MASK(64))' can't fail
if 'dev->dma_mask' is non-NULL. And if it is NULL, it would fail for the
same reason when tried with DMA_BIT_MASK(32).
[1]: https://lkml.org/lkml/2021/6/7/398
Fixes: ecb8c88bd3
("dmaengine: dw-edma-pcie: switch from 'pci_' to 'dma_' API")
Signed-off-by: Christophe JAILLET <christophe.jaillet@wanadoo.fr>
Link: https://lore.kernel.org/r/935fbb40ae930c5fe87482a41dcb73abf2257973.1636492127.git.christophe.jaillet@wanadoo.fr
Signed-off-by: Vinod Koul <vkoul@kernel.org>
380 lines
11 KiB
C
380 lines
11 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Copyright (c) 2018-2019 Synopsys, Inc. and/or its affiliates.
|
|
* Synopsys DesignWare eDMA PCIe driver
|
|
*
|
|
* Author: Gustavo Pimentel <gustavo.pimentel@synopsys.com>
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/module.h>
|
|
#include <linux/pci.h>
|
|
#include <linux/device.h>
|
|
#include <linux/dma/edma.h>
|
|
#include <linux/pci-epf.h>
|
|
#include <linux/msi.h>
|
|
#include <linux/bitfield.h>
|
|
|
|
#include "dw-edma-core.h"
|
|
|
|
#define DW_PCIE_VSEC_DMA_ID 0x6
|
|
#define DW_PCIE_VSEC_DMA_BAR GENMASK(10, 8)
|
|
#define DW_PCIE_VSEC_DMA_MAP GENMASK(2, 0)
|
|
#define DW_PCIE_VSEC_DMA_WR_CH GENMASK(9, 0)
|
|
#define DW_PCIE_VSEC_DMA_RD_CH GENMASK(25, 16)
|
|
|
|
#define DW_BLOCK(a, b, c) \
|
|
{ \
|
|
.bar = a, \
|
|
.off = b, \
|
|
.sz = c, \
|
|
},
|
|
|
|
struct dw_edma_block {
|
|
enum pci_barno bar;
|
|
off_t off;
|
|
size_t sz;
|
|
};
|
|
|
|
struct dw_edma_pcie_data {
|
|
/* eDMA registers location */
|
|
struct dw_edma_block rg;
|
|
/* eDMA memory linked list location */
|
|
struct dw_edma_block ll_wr[EDMA_MAX_WR_CH];
|
|
struct dw_edma_block ll_rd[EDMA_MAX_RD_CH];
|
|
/* eDMA memory data location */
|
|
struct dw_edma_block dt_wr[EDMA_MAX_WR_CH];
|
|
struct dw_edma_block dt_rd[EDMA_MAX_RD_CH];
|
|
/* Other */
|
|
enum dw_edma_map_format mf;
|
|
u8 irqs;
|
|
u16 wr_ch_cnt;
|
|
u16 rd_ch_cnt;
|
|
};
|
|
|
|
static const struct dw_edma_pcie_data snps_edda_data = {
|
|
/* eDMA registers location */
|
|
.rg.bar = BAR_0,
|
|
.rg.off = 0x00001000, /* 4 Kbytes */
|
|
.rg.sz = 0x00002000, /* 8 Kbytes */
|
|
/* eDMA memory linked list location */
|
|
.ll_wr = {
|
|
/* Channel 0 - BAR 2, offset 0 Mbytes, size 2 Kbytes */
|
|
DW_BLOCK(BAR_2, 0x00000000, 0x00000800)
|
|
/* Channel 1 - BAR 2, offset 2 Mbytes, size 2 Kbytes */
|
|
DW_BLOCK(BAR_2, 0x00200000, 0x00000800)
|
|
},
|
|
.ll_rd = {
|
|
/* Channel 0 - BAR 2, offset 4 Mbytes, size 2 Kbytes */
|
|
DW_BLOCK(BAR_2, 0x00400000, 0x00000800)
|
|
/* Channel 1 - BAR 2, offset 6 Mbytes, size 2 Kbytes */
|
|
DW_BLOCK(BAR_2, 0x00600000, 0x00000800)
|
|
},
|
|
/* eDMA memory data location */
|
|
.dt_wr = {
|
|
/* Channel 0 - BAR 2, offset 8 Mbytes, size 2 Kbytes */
|
|
DW_BLOCK(BAR_2, 0x00800000, 0x00000800)
|
|
/* Channel 1 - BAR 2, offset 9 Mbytes, size 2 Kbytes */
|
|
DW_BLOCK(BAR_2, 0x00900000, 0x00000800)
|
|
},
|
|
.dt_rd = {
|
|
/* Channel 0 - BAR 2, offset 10 Mbytes, size 2 Kbytes */
|
|
DW_BLOCK(BAR_2, 0x00a00000, 0x00000800)
|
|
/* Channel 1 - BAR 2, offset 11 Mbytes, size 2 Kbytes */
|
|
DW_BLOCK(BAR_2, 0x00b00000, 0x00000800)
|
|
},
|
|
/* Other */
|
|
.mf = EDMA_MF_EDMA_UNROLL,
|
|
.irqs = 1,
|
|
.wr_ch_cnt = 2,
|
|
.rd_ch_cnt = 2,
|
|
};
|
|
|
|
static int dw_edma_pcie_irq_vector(struct device *dev, unsigned int nr)
|
|
{
|
|
return pci_irq_vector(to_pci_dev(dev), nr);
|
|
}
|
|
|
|
static const struct dw_edma_core_ops dw_edma_pcie_core_ops = {
|
|
.irq_vector = dw_edma_pcie_irq_vector,
|
|
};
|
|
|
|
static void dw_edma_pcie_get_vsec_dma_data(struct pci_dev *pdev,
|
|
struct dw_edma_pcie_data *pdata)
|
|
{
|
|
u32 val, map;
|
|
u16 vsec;
|
|
u64 off;
|
|
|
|
vsec = pci_find_vsec_capability(pdev, PCI_VENDOR_ID_SYNOPSYS,
|
|
DW_PCIE_VSEC_DMA_ID);
|
|
if (!vsec)
|
|
return;
|
|
|
|
pci_read_config_dword(pdev, vsec + PCI_VNDR_HEADER, &val);
|
|
if (PCI_VNDR_HEADER_REV(val) != 0x00 ||
|
|
PCI_VNDR_HEADER_LEN(val) != 0x18)
|
|
return;
|
|
|
|
pci_dbg(pdev, "Detected PCIe Vendor-Specific Extended Capability DMA\n");
|
|
pci_read_config_dword(pdev, vsec + 0x8, &val);
|
|
map = FIELD_GET(DW_PCIE_VSEC_DMA_MAP, val);
|
|
if (map != EDMA_MF_EDMA_LEGACY &&
|
|
map != EDMA_MF_EDMA_UNROLL &&
|
|
map != EDMA_MF_HDMA_COMPAT)
|
|
return;
|
|
|
|
pdata->mf = map;
|
|
pdata->rg.bar = FIELD_GET(DW_PCIE_VSEC_DMA_BAR, val);
|
|
|
|
pci_read_config_dword(pdev, vsec + 0xc, &val);
|
|
pdata->wr_ch_cnt = min_t(u16, pdata->wr_ch_cnt,
|
|
FIELD_GET(DW_PCIE_VSEC_DMA_WR_CH, val));
|
|
pdata->rd_ch_cnt = min_t(u16, pdata->rd_ch_cnt,
|
|
FIELD_GET(DW_PCIE_VSEC_DMA_RD_CH, val));
|
|
|
|
pci_read_config_dword(pdev, vsec + 0x14, &val);
|
|
off = val;
|
|
pci_read_config_dword(pdev, vsec + 0x10, &val);
|
|
off <<= 32;
|
|
off |= val;
|
|
pdata->rg.off = off;
|
|
}
|
|
|
|
static int dw_edma_pcie_probe(struct pci_dev *pdev,
|
|
const struct pci_device_id *pid)
|
|
{
|
|
struct dw_edma_pcie_data *pdata = (void *)pid->driver_data;
|
|
struct dw_edma_pcie_data vsec_data;
|
|
struct device *dev = &pdev->dev;
|
|
struct dw_edma_chip *chip;
|
|
struct dw_edma *dw;
|
|
int err, nr_irqs;
|
|
int i, mask;
|
|
|
|
/* Enable PCI device */
|
|
err = pcim_enable_device(pdev);
|
|
if (err) {
|
|
pci_err(pdev, "enabling device failed\n");
|
|
return err;
|
|
}
|
|
|
|
memcpy(&vsec_data, pdata, sizeof(struct dw_edma_pcie_data));
|
|
|
|
/*
|
|
* Tries to find if exists a PCIe Vendor-Specific Extended Capability
|
|
* for the DMA, if one exists, then reconfigures it.
|
|
*/
|
|
dw_edma_pcie_get_vsec_dma_data(pdev, &vsec_data);
|
|
|
|
/* Mapping PCI BAR regions */
|
|
mask = BIT(vsec_data.rg.bar);
|
|
for (i = 0; i < vsec_data.wr_ch_cnt; i++) {
|
|
mask |= BIT(vsec_data.ll_wr[i].bar);
|
|
mask |= BIT(vsec_data.dt_wr[i].bar);
|
|
}
|
|
for (i = 0; i < vsec_data.rd_ch_cnt; i++) {
|
|
mask |= BIT(vsec_data.ll_rd[i].bar);
|
|
mask |= BIT(vsec_data.dt_rd[i].bar);
|
|
}
|
|
err = pcim_iomap_regions(pdev, mask, pci_name(pdev));
|
|
if (err) {
|
|
pci_err(pdev, "eDMA BAR I/O remapping failed\n");
|
|
return err;
|
|
}
|
|
|
|
pci_set_master(pdev);
|
|
|
|
/* DMA configuration */
|
|
err = dma_set_mask_and_coherent(&pdev->dev, DMA_BIT_MASK(64));
|
|
if (err) {
|
|
pci_err(pdev, "DMA mask 64 set failed\n");
|
|
return err;
|
|
}
|
|
|
|
/* Data structure allocation */
|
|
chip = devm_kzalloc(dev, sizeof(*chip), GFP_KERNEL);
|
|
if (!chip)
|
|
return -ENOMEM;
|
|
|
|
dw = devm_kzalloc(dev, sizeof(*dw), GFP_KERNEL);
|
|
if (!dw)
|
|
return -ENOMEM;
|
|
|
|
/* IRQs allocation */
|
|
nr_irqs = pci_alloc_irq_vectors(pdev, 1, vsec_data.irqs,
|
|
PCI_IRQ_MSI | PCI_IRQ_MSIX);
|
|
if (nr_irqs < 1) {
|
|
pci_err(pdev, "fail to alloc IRQ vector (number of IRQs=%u)\n",
|
|
nr_irqs);
|
|
return -EPERM;
|
|
}
|
|
|
|
/* Data structure initialization */
|
|
chip->dw = dw;
|
|
chip->dev = dev;
|
|
chip->id = pdev->devfn;
|
|
chip->irq = pdev->irq;
|
|
|
|
dw->mf = vsec_data.mf;
|
|
dw->nr_irqs = nr_irqs;
|
|
dw->ops = &dw_edma_pcie_core_ops;
|
|
dw->wr_ch_cnt = vsec_data.wr_ch_cnt;
|
|
dw->rd_ch_cnt = vsec_data.rd_ch_cnt;
|
|
|
|
dw->rg_region.vaddr = pcim_iomap_table(pdev)[vsec_data.rg.bar];
|
|
if (!dw->rg_region.vaddr)
|
|
return -ENOMEM;
|
|
|
|
dw->rg_region.vaddr += vsec_data.rg.off;
|
|
dw->rg_region.paddr = pdev->resource[vsec_data.rg.bar].start;
|
|
dw->rg_region.paddr += vsec_data.rg.off;
|
|
dw->rg_region.sz = vsec_data.rg.sz;
|
|
|
|
for (i = 0; i < dw->wr_ch_cnt; i++) {
|
|
struct dw_edma_region *ll_region = &dw->ll_region_wr[i];
|
|
struct dw_edma_region *dt_region = &dw->dt_region_wr[i];
|
|
struct dw_edma_block *ll_block = &vsec_data.ll_wr[i];
|
|
struct dw_edma_block *dt_block = &vsec_data.dt_wr[i];
|
|
|
|
ll_region->vaddr = pcim_iomap_table(pdev)[ll_block->bar];
|
|
if (!ll_region->vaddr)
|
|
return -ENOMEM;
|
|
|
|
ll_region->vaddr += ll_block->off;
|
|
ll_region->paddr = pdev->resource[ll_block->bar].start;
|
|
ll_region->paddr += ll_block->off;
|
|
ll_region->sz = ll_block->sz;
|
|
|
|
dt_region->vaddr = pcim_iomap_table(pdev)[dt_block->bar];
|
|
if (!dt_region->vaddr)
|
|
return -ENOMEM;
|
|
|
|
dt_region->vaddr += dt_block->off;
|
|
dt_region->paddr = pdev->resource[dt_block->bar].start;
|
|
dt_region->paddr += dt_block->off;
|
|
dt_region->sz = dt_block->sz;
|
|
}
|
|
|
|
for (i = 0; i < dw->rd_ch_cnt; i++) {
|
|
struct dw_edma_region *ll_region = &dw->ll_region_rd[i];
|
|
struct dw_edma_region *dt_region = &dw->dt_region_rd[i];
|
|
struct dw_edma_block *ll_block = &vsec_data.ll_rd[i];
|
|
struct dw_edma_block *dt_block = &vsec_data.dt_rd[i];
|
|
|
|
ll_region->vaddr = pcim_iomap_table(pdev)[ll_block->bar];
|
|
if (!ll_region->vaddr)
|
|
return -ENOMEM;
|
|
|
|
ll_region->vaddr += ll_block->off;
|
|
ll_region->paddr = pdev->resource[ll_block->bar].start;
|
|
ll_region->paddr += ll_block->off;
|
|
ll_region->sz = ll_block->sz;
|
|
|
|
dt_region->vaddr = pcim_iomap_table(pdev)[dt_block->bar];
|
|
if (!dt_region->vaddr)
|
|
return -ENOMEM;
|
|
|
|
dt_region->vaddr += dt_block->off;
|
|
dt_region->paddr = pdev->resource[dt_block->bar].start;
|
|
dt_region->paddr += dt_block->off;
|
|
dt_region->sz = dt_block->sz;
|
|
}
|
|
|
|
/* Debug info */
|
|
if (dw->mf == EDMA_MF_EDMA_LEGACY)
|
|
pci_dbg(pdev, "Version:\teDMA Port Logic (0x%x)\n", dw->mf);
|
|
else if (dw->mf == EDMA_MF_EDMA_UNROLL)
|
|
pci_dbg(pdev, "Version:\teDMA Unroll (0x%x)\n", dw->mf);
|
|
else if (dw->mf == EDMA_MF_HDMA_COMPAT)
|
|
pci_dbg(pdev, "Version:\tHDMA Compatible (0x%x)\n", dw->mf);
|
|
else
|
|
pci_dbg(pdev, "Version:\tUnknown (0x%x)\n", dw->mf);
|
|
|
|
pci_dbg(pdev, "Registers:\tBAR=%u, off=0x%.8lx, sz=0x%zx bytes, addr(v=%p, p=%pa)\n",
|
|
vsec_data.rg.bar, vsec_data.rg.off, vsec_data.rg.sz,
|
|
dw->rg_region.vaddr, &dw->rg_region.paddr);
|
|
|
|
|
|
for (i = 0; i < dw->wr_ch_cnt; i++) {
|
|
pci_dbg(pdev, "L. List:\tWRITE CH%.2u, BAR=%u, off=0x%.8lx, sz=0x%zx bytes, addr(v=%p, p=%pa)\n",
|
|
i, vsec_data.ll_wr[i].bar,
|
|
vsec_data.ll_wr[i].off, dw->ll_region_wr[i].sz,
|
|
dw->ll_region_wr[i].vaddr, &dw->ll_region_wr[i].paddr);
|
|
|
|
pci_dbg(pdev, "Data:\tWRITE CH%.2u, BAR=%u, off=0x%.8lx, sz=0x%zx bytes, addr(v=%p, p=%pa)\n",
|
|
i, vsec_data.dt_wr[i].bar,
|
|
vsec_data.dt_wr[i].off, dw->dt_region_wr[i].sz,
|
|
dw->dt_region_wr[i].vaddr, &dw->dt_region_wr[i].paddr);
|
|
}
|
|
|
|
for (i = 0; i < dw->rd_ch_cnt; i++) {
|
|
pci_dbg(pdev, "L. List:\tREAD CH%.2u, BAR=%u, off=0x%.8lx, sz=0x%zx bytes, addr(v=%p, p=%pa)\n",
|
|
i, vsec_data.ll_rd[i].bar,
|
|
vsec_data.ll_rd[i].off, dw->ll_region_rd[i].sz,
|
|
dw->ll_region_rd[i].vaddr, &dw->ll_region_rd[i].paddr);
|
|
|
|
pci_dbg(pdev, "Data:\tREAD CH%.2u, BAR=%u, off=0x%.8lx, sz=0x%zx bytes, addr(v=%p, p=%pa)\n",
|
|
i, vsec_data.dt_rd[i].bar,
|
|
vsec_data.dt_rd[i].off, dw->dt_region_rd[i].sz,
|
|
dw->dt_region_rd[i].vaddr, &dw->dt_region_rd[i].paddr);
|
|
}
|
|
|
|
pci_dbg(pdev, "Nr. IRQs:\t%u\n", dw->nr_irqs);
|
|
|
|
/* Validating if PCI interrupts were enabled */
|
|
if (!pci_dev_msi_enabled(pdev)) {
|
|
pci_err(pdev, "enable interrupt failed\n");
|
|
return -EPERM;
|
|
}
|
|
|
|
dw->irq = devm_kcalloc(dev, nr_irqs, sizeof(*dw->irq), GFP_KERNEL);
|
|
if (!dw->irq)
|
|
return -ENOMEM;
|
|
|
|
/* Starting eDMA driver */
|
|
err = dw_edma_probe(chip);
|
|
if (err) {
|
|
pci_err(pdev, "eDMA probe failed\n");
|
|
return err;
|
|
}
|
|
|
|
/* Saving data structure reference */
|
|
pci_set_drvdata(pdev, chip);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void dw_edma_pcie_remove(struct pci_dev *pdev)
|
|
{
|
|
struct dw_edma_chip *chip = pci_get_drvdata(pdev);
|
|
int err;
|
|
|
|
/* Stopping eDMA driver */
|
|
err = dw_edma_remove(chip);
|
|
if (err)
|
|
pci_warn(pdev, "can't remove device properly: %d\n", err);
|
|
|
|
/* Freeing IRQs */
|
|
pci_free_irq_vectors(pdev);
|
|
}
|
|
|
|
static const struct pci_device_id dw_edma_pcie_id_table[] = {
|
|
{ PCI_DEVICE_DATA(SYNOPSYS, EDDA, &snps_edda_data) },
|
|
{ }
|
|
};
|
|
MODULE_DEVICE_TABLE(pci, dw_edma_pcie_id_table);
|
|
|
|
static struct pci_driver dw_edma_pcie_driver = {
|
|
.name = "dw-edma-pcie",
|
|
.id_table = dw_edma_pcie_id_table,
|
|
.probe = dw_edma_pcie_probe,
|
|
.remove = dw_edma_pcie_remove,
|
|
};
|
|
|
|
module_pci_driver(dw_edma_pcie_driver);
|
|
|
|
MODULE_LICENSE("GPL v2");
|
|
MODULE_DESCRIPTION("Synopsys DesignWare eDMA PCIe driver");
|
|
MODULE_AUTHOR("Gustavo Pimentel <gustavo.pimentel@synopsys.com>");
|