421 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			421 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // 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>");
 | |
| 
 |