|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | /* | 
|  | * Copyright (c) 2020 Synopsys, Inc. and/or its affiliates. | 
|  | * Synopsys DesignWare xData driver | 
|  | * | 
|  | * Author: Gustavo Pimentel <gustavo.pimentel@synopsys.com> | 
|  | */ | 
|  |  | 
|  | #include <linux/miscdevice.h> | 
|  | #include <linux/bitfield.h> | 
|  | #include <linux/pci-epf.h> | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/device.h> | 
|  | #include <linux/bitops.h> | 
|  | #include <linux/mutex.h> | 
|  | #include <linux/delay.h> | 
|  | #include <linux/pci.h> | 
|  |  | 
|  | #define DW_XDATA_DRIVER_NAME		"dw-xdata-pcie" | 
|  |  | 
|  | #define DW_XDATA_EP_MEM_OFFSET		0x8000000 | 
|  |  | 
|  | static DEFINE_IDA(xdata_ida); | 
|  |  | 
|  | #define STATUS_DONE			BIT(0) | 
|  |  | 
|  | #define CONTROL_DOORBELL		BIT(0) | 
|  | #define CONTROL_IS_WRITE		BIT(1) | 
|  | #define CONTROL_LENGTH(a)		FIELD_PREP(GENMASK(13, 2), a) | 
|  | #define CONTROL_PATTERN_INC		BIT(16) | 
|  | #define CONTROL_NO_ADDR_INC		BIT(18) | 
|  |  | 
|  | #define XPERF_CONTROL_ENABLE		BIT(5) | 
|  |  | 
|  | #define BURST_REPEAT			BIT(31) | 
|  | #define BURST_VALUE			0x1001 | 
|  |  | 
|  | #define PATTERN_VALUE			0x0 | 
|  |  | 
|  | struct dw_xdata_regs { | 
|  | u32 addr_lsb;					/* 0x000 */ | 
|  | u32 addr_msb;					/* 0x004 */ | 
|  | u32 burst_cnt;					/* 0x008 */ | 
|  | u32 control;					/* 0x00c */ | 
|  | u32 pattern;					/* 0x010 */ | 
|  | u32 status;					/* 0x014 */ | 
|  | u32 RAM_addr;					/* 0x018 */ | 
|  | u32 RAM_port;					/* 0x01c */ | 
|  | u32 _reserved0[14];				/* 0x020..0x054 */ | 
|  | u32 perf_control;				/* 0x058 */ | 
|  | u32 _reserved1[41];				/* 0x05c..0x0fc */ | 
|  | u32 wr_cnt_lsb;					/* 0x100 */ | 
|  | u32 wr_cnt_msb;					/* 0x104 */ | 
|  | u32 rd_cnt_lsb;					/* 0x108 */ | 
|  | u32 rd_cnt_msb;					/* 0x10c */ | 
|  | } __packed; | 
|  |  | 
|  | struct dw_xdata_region { | 
|  | phys_addr_t paddr;				/* physical address */ | 
|  | void __iomem *vaddr;				/* virtual address */ | 
|  | }; | 
|  |  | 
|  | struct dw_xdata { | 
|  | struct dw_xdata_region rg_region;		/* registers */ | 
|  | size_t max_wr_len;				/* max wr xfer len */ | 
|  | size_t max_rd_len;				/* max rd xfer len */ | 
|  | struct mutex mutex; | 
|  | struct pci_dev *pdev; | 
|  | struct miscdevice misc_dev; | 
|  | }; | 
|  |  | 
|  | static inline struct dw_xdata_regs __iomem *__dw_regs(struct dw_xdata *dw) | 
|  | { | 
|  | return dw->rg_region.vaddr; | 
|  | } | 
|  |  | 
|  | static void dw_xdata_stop(struct dw_xdata *dw) | 
|  | { | 
|  | u32 burst; | 
|  |  | 
|  | mutex_lock(&dw->mutex); | 
|  |  | 
|  | burst = readl(&(__dw_regs(dw)->burst_cnt)); | 
|  |  | 
|  | if (burst & BURST_REPEAT) { | 
|  | burst &= ~(u32)BURST_REPEAT; | 
|  | writel(burst, &(__dw_regs(dw)->burst_cnt)); | 
|  | } | 
|  |  | 
|  | mutex_unlock(&dw->mutex); | 
|  | } | 
|  |  | 
|  | static void dw_xdata_start(struct dw_xdata *dw, bool write) | 
|  | { | 
|  | struct device *dev = &dw->pdev->dev; | 
|  | u32 control, status; | 
|  |  | 
|  | /* Stop first if xfer in progress */ | 
|  | dw_xdata_stop(dw); | 
|  |  | 
|  | mutex_lock(&dw->mutex); | 
|  |  | 
|  | /* Clear status register */ | 
|  | writel(0x0, &(__dw_regs(dw)->status)); | 
|  |  | 
|  | /* Burst count register set for continuous until stopped */ | 
|  | writel(BURST_REPEAT | BURST_VALUE, &(__dw_regs(dw)->burst_cnt)); | 
|  |  | 
|  | /* Pattern register */ | 
|  | writel(PATTERN_VALUE, &(__dw_regs(dw)->pattern)); | 
|  |  | 
|  | /* Control register */ | 
|  | control = CONTROL_DOORBELL | CONTROL_PATTERN_INC | CONTROL_NO_ADDR_INC; | 
|  | if (write) { | 
|  | control |= CONTROL_IS_WRITE; | 
|  | control |= CONTROL_LENGTH(dw->max_wr_len); | 
|  | } else { | 
|  | control |= CONTROL_LENGTH(dw->max_rd_len); | 
|  | } | 
|  | writel(control, &(__dw_regs(dw)->control)); | 
|  |  | 
|  | /* | 
|  | * The xData HW block needs about 100 ms to initiate the traffic | 
|  | * generation according this HW block datasheet. | 
|  | */ | 
|  | usleep_range(100, 150); | 
|  |  | 
|  | status = readl(&(__dw_regs(dw)->status)); | 
|  |  | 
|  | mutex_unlock(&dw->mutex); | 
|  |  | 
|  | if (!(status & STATUS_DONE)) | 
|  | dev_dbg(dev, "xData: started %s direction\n", | 
|  | write ? "write" : "read"); | 
|  | } | 
|  |  | 
|  | static void dw_xdata_perf_meas(struct dw_xdata *dw, u64 *data, bool write) | 
|  | { | 
|  | if (write) { | 
|  | *data = readl(&(__dw_regs(dw)->wr_cnt_msb)); | 
|  | *data <<= 32; | 
|  | *data |= readl(&(__dw_regs(dw)->wr_cnt_lsb)); | 
|  | } else { | 
|  | *data = readl(&(__dw_regs(dw)->rd_cnt_msb)); | 
|  | *data <<= 32; | 
|  | *data |= readl(&(__dw_regs(dw)->rd_cnt_lsb)); | 
|  | } | 
|  | } | 
|  |  | 
|  | static u64 dw_xdata_perf_diff(u64 *m1, u64 *m2, u64 time) | 
|  | { | 
|  | u64 rate = (*m1 - *m2); | 
|  |  | 
|  | rate *= (1000 * 1000 * 1000); | 
|  | rate >>= 20; | 
|  | rate = DIV_ROUND_CLOSEST_ULL(rate, time); | 
|  |  | 
|  | return rate; | 
|  | } | 
|  |  | 
|  | static void dw_xdata_perf(struct dw_xdata *dw, u64 *rate, bool write) | 
|  | { | 
|  | struct device *dev = &dw->pdev->dev; | 
|  | u64 data[2], time[2], diff; | 
|  |  | 
|  | mutex_lock(&dw->mutex); | 
|  |  | 
|  | /* First acquisition of current count frames */ | 
|  | writel(0x0, &(__dw_regs(dw)->perf_control)); | 
|  | dw_xdata_perf_meas(dw, &data[0], write); | 
|  | time[0] = jiffies; | 
|  | writel((u32)XPERF_CONTROL_ENABLE, &(__dw_regs(dw)->perf_control)); | 
|  |  | 
|  | /* | 
|  | * Wait 100ms between the 1st count frame acquisition and the 2nd | 
|  | * count frame acquisition, in order to calculate the speed later | 
|  | */ | 
|  | mdelay(100); | 
|  |  | 
|  | /* Second acquisition of current count frames */ | 
|  | writel(0x0, &(__dw_regs(dw)->perf_control)); | 
|  | dw_xdata_perf_meas(dw, &data[1], write); | 
|  | time[1] = jiffies; | 
|  | writel((u32)XPERF_CONTROL_ENABLE, &(__dw_regs(dw)->perf_control)); | 
|  |  | 
|  | /* | 
|  | * Speed calculation | 
|  | * | 
|  | * rate = (2nd count frames - 1st count frames) / (time elapsed) | 
|  | */ | 
|  | diff = jiffies_to_nsecs(time[1] - time[0]); | 
|  | *rate = dw_xdata_perf_diff(&data[1], &data[0], diff); | 
|  |  | 
|  | mutex_unlock(&dw->mutex); | 
|  |  | 
|  | dev_dbg(dev, "xData: time=%llu us, %s=%llu MB/s\n", | 
|  | diff, write ? "write" : "read", *rate); | 
|  | } | 
|  |  | 
|  | static struct dw_xdata *misc_dev_to_dw(struct miscdevice *misc_dev) | 
|  | { | 
|  | return container_of(misc_dev, struct dw_xdata, misc_dev); | 
|  | } | 
|  |  | 
|  | static ssize_t write_show(struct device *dev, struct device_attribute *attr, | 
|  | char *buf) | 
|  | { | 
|  | struct miscdevice *misc_dev = dev_get_drvdata(dev); | 
|  | struct dw_xdata *dw = misc_dev_to_dw(misc_dev); | 
|  | u64 rate; | 
|  |  | 
|  | dw_xdata_perf(dw, &rate, true); | 
|  |  | 
|  | return sysfs_emit(buf, "%llu\n", rate); | 
|  | } | 
|  |  | 
|  | static ssize_t write_store(struct device *dev, struct device_attribute *attr, | 
|  | const char *buf, size_t size) | 
|  | { | 
|  | struct miscdevice *misc_dev = dev_get_drvdata(dev); | 
|  | struct dw_xdata *dw = misc_dev_to_dw(misc_dev); | 
|  | bool enabled; | 
|  | int ret; | 
|  |  | 
|  | ret = kstrtobool(buf, &enabled); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (enabled) { | 
|  | dev_dbg(dev, "xData: requested write transfer\n"); | 
|  | dw_xdata_start(dw, true); | 
|  | } else { | 
|  | dev_dbg(dev, "xData: requested stop transfer\n"); | 
|  | dw_xdata_stop(dw); | 
|  | } | 
|  |  | 
|  | return size; | 
|  | } | 
|  |  | 
|  | static DEVICE_ATTR_RW(write); | 
|  |  | 
|  | static ssize_t read_show(struct device *dev, struct device_attribute *attr, | 
|  | char *buf) | 
|  | { | 
|  | struct miscdevice *misc_dev = dev_get_drvdata(dev); | 
|  | struct dw_xdata *dw = misc_dev_to_dw(misc_dev); | 
|  | u64 rate; | 
|  |  | 
|  | dw_xdata_perf(dw, &rate, false); | 
|  |  | 
|  | return sysfs_emit(buf, "%llu\n", rate); | 
|  | } | 
|  |  | 
|  | static ssize_t read_store(struct device *dev, struct device_attribute *attr, | 
|  | const char *buf, size_t size) | 
|  | { | 
|  | struct miscdevice *misc_dev = dev_get_drvdata(dev); | 
|  | struct dw_xdata *dw = misc_dev_to_dw(misc_dev); | 
|  | bool enabled; | 
|  | int ret; | 
|  |  | 
|  | ret = kstrtobool(buf, &enabled); | 
|  | if (ret < 0) | 
|  | return ret; | 
|  |  | 
|  | if (enabled) { | 
|  | dev_dbg(dev, "xData: requested read transfer\n"); | 
|  | dw_xdata_start(dw, false); | 
|  | } else { | 
|  | dev_dbg(dev, "xData: requested stop transfer\n"); | 
|  | dw_xdata_stop(dw); | 
|  | } | 
|  |  | 
|  | return size; | 
|  | } | 
|  |  | 
|  | static DEVICE_ATTR_RW(read); | 
|  |  | 
|  | static struct attribute *xdata_attrs[] = { | 
|  | &dev_attr_write.attr, | 
|  | &dev_attr_read.attr, | 
|  | NULL, | 
|  | }; | 
|  |  | 
|  | ATTRIBUTE_GROUPS(xdata); | 
|  |  | 
|  | static int dw_xdata_pcie_probe(struct pci_dev *pdev, | 
|  | const struct pci_device_id *pid) | 
|  | { | 
|  | struct device *dev = &pdev->dev; | 
|  | struct dw_xdata *dw; | 
|  | char name[24]; | 
|  | u64 addr; | 
|  | int err; | 
|  | int id; | 
|  |  | 
|  | /* Enable PCI device */ | 
|  | err = pcim_enable_device(pdev); | 
|  | if (err) { | 
|  | dev_err(dev, "enabling device failed\n"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | /* Mapping PCI BAR regions */ | 
|  | err = pcim_iomap_regions(pdev, BIT(BAR_0), pci_name(pdev)); | 
|  | if (err) { | 
|  | dev_err(dev, "xData BAR I/O remapping failed\n"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | pci_set_master(pdev); | 
|  |  | 
|  | /* Allocate memory */ | 
|  | dw = devm_kzalloc(dev, sizeof(*dw), GFP_KERNEL); | 
|  | if (!dw) | 
|  | return -ENOMEM; | 
|  |  | 
|  | /* Data structure initialization */ | 
|  | mutex_init(&dw->mutex); | 
|  |  | 
|  | dw->rg_region.vaddr = pcim_iomap_table(pdev)[BAR_0]; | 
|  | if (!dw->rg_region.vaddr) | 
|  | return -ENOMEM; | 
|  |  | 
|  | dw->rg_region.paddr = pdev->resource[BAR_0].start; | 
|  |  | 
|  | dw->max_wr_len = pcie_get_mps(pdev); | 
|  | dw->max_wr_len >>= 2; | 
|  |  | 
|  | dw->max_rd_len = pcie_get_readrq(pdev); | 
|  | dw->max_rd_len >>= 2; | 
|  |  | 
|  | dw->pdev = pdev; | 
|  |  | 
|  | id = ida_alloc(&xdata_ida, GFP_KERNEL); | 
|  | if (id < 0) { | 
|  | dev_err(dev, "xData: unable to get id\n"); | 
|  | return id; | 
|  | } | 
|  |  | 
|  | snprintf(name, sizeof(name), DW_XDATA_DRIVER_NAME ".%d", id); | 
|  | dw->misc_dev.name = kstrdup(name, GFP_KERNEL); | 
|  | if (!dw->misc_dev.name) { | 
|  | err = -ENOMEM; | 
|  | goto err_ida_remove; | 
|  | } | 
|  |  | 
|  | dw->misc_dev.minor = MISC_DYNAMIC_MINOR; | 
|  | dw->misc_dev.parent = dev; | 
|  | dw->misc_dev.groups = xdata_groups; | 
|  |  | 
|  | writel(0x0, &(__dw_regs(dw)->RAM_addr)); | 
|  | writel(0x0, &(__dw_regs(dw)->RAM_port)); | 
|  |  | 
|  | addr = dw->rg_region.paddr + DW_XDATA_EP_MEM_OFFSET; | 
|  | writel(lower_32_bits(addr), &(__dw_regs(dw)->addr_lsb)); | 
|  | writel(upper_32_bits(addr), &(__dw_regs(dw)->addr_msb)); | 
|  | dev_dbg(dev, "xData: target address = 0x%.16llx\n", addr); | 
|  |  | 
|  | dev_dbg(dev, "xData: wr_len = %zu, rd_len = %zu\n", | 
|  | dw->max_wr_len * 4, dw->max_rd_len * 4); | 
|  |  | 
|  | /* Saving data structure reference */ | 
|  | pci_set_drvdata(pdev, dw); | 
|  |  | 
|  | /* Register misc device */ | 
|  | err = misc_register(&dw->misc_dev); | 
|  | if (err) { | 
|  | dev_err(dev, "xData: failed to register device\n"); | 
|  | goto err_kfree_name; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  |  | 
|  | err_kfree_name: | 
|  | kfree(dw->misc_dev.name); | 
|  |  | 
|  | err_ida_remove: | 
|  | ida_free(&xdata_ida, id); | 
|  |  | 
|  | return err; | 
|  | } | 
|  |  | 
|  | static void dw_xdata_pcie_remove(struct pci_dev *pdev) | 
|  | { | 
|  | struct dw_xdata *dw = pci_get_drvdata(pdev); | 
|  | int id; | 
|  |  | 
|  | if (sscanf(dw->misc_dev.name, DW_XDATA_DRIVER_NAME ".%d", &id) != 1) | 
|  | return; | 
|  |  | 
|  | if (id < 0) | 
|  | return; | 
|  |  | 
|  | dw_xdata_stop(dw); | 
|  | misc_deregister(&dw->misc_dev); | 
|  | kfree(dw->misc_dev.name); | 
|  | ida_free(&xdata_ida, id); | 
|  | } | 
|  |  | 
|  | static const struct pci_device_id dw_xdata_pcie_id_table[] = { | 
|  | { PCI_DEVICE_DATA(SYNOPSYS, EDDA, NULL) }, | 
|  | { } | 
|  | }; | 
|  | MODULE_DEVICE_TABLE(pci, dw_xdata_pcie_id_table); | 
|  |  | 
|  | static struct pci_driver dw_xdata_pcie_driver = { | 
|  | .name		= DW_XDATA_DRIVER_NAME, | 
|  | .id_table	= dw_xdata_pcie_id_table, | 
|  | .probe		= dw_xdata_pcie_probe, | 
|  | .remove		= dw_xdata_pcie_remove, | 
|  | }; | 
|  |  | 
|  | module_pci_driver(dw_xdata_pcie_driver); | 
|  |  | 
|  | MODULE_LICENSE("GPL v2"); | 
|  | MODULE_DESCRIPTION("Synopsys DesignWare xData PCIe driver"); | 
|  | MODULE_AUTHOR("Gustavo Pimentel <gustavo.pimentel@synopsys.com>"); | 
|  |  |