misc: Add Synopsys DesignWare xData IP driver
Add Synopsys DesignWare xData IP driver. This driver enables/disables the PCI traffic generator module pertain to the Synopsys DesignWare prototype. Signed-off-by: Gustavo Pimentel <gustavo.pimentel@synopsys.com> Link: https://lore.kernel.org/r/daa1efe23850e77d6807dc3f371728fc0b7548b8.1617016509.git.gustavo.pimentel@synopsys.com Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
		
							parent
							
								
									b2192cfeba
								
							
						
					
					
						commit
						e8a30eef6e
					
				| @ -402,6 +402,16 @@ config SRAM | ||||
| config SRAM_EXEC | ||||
| 	bool | ||||
| 
 | ||||
| config DW_XDATA_PCIE | ||||
| 	depends on PCI | ||||
| 	tristate "Synopsys DesignWare xData PCIe driver" | ||||
| 	help | ||||
| 	  This driver allows controlling Synopsys DesignWare PCIe traffic | ||||
| 	  generator IP also known as xData, present in Synopsys DesignWare | ||||
| 	  PCIe Endpoint prototype. | ||||
| 
 | ||||
| 	  If unsure, say N. | ||||
| 
 | ||||
| config PCI_ENDPOINT_TEST | ||||
| 	depends on PCI | ||||
| 	select CRC32 | ||||
|  | ||||
| @ -47,6 +47,7 @@ obj-$(CONFIG_SRAM_EXEC)		+= sram-exec.o | ||||
| obj-$(CONFIG_GENWQE)		+= genwqe/ | ||||
| obj-$(CONFIG_ECHO)		+= echo/ | ||||
| obj-$(CONFIG_CXL_BASE)		+= cxl/ | ||||
| obj-$(CONFIG_DW_XDATA_PCIE)	+= dw-xdata-pcie.o | ||||
| obj-$(CONFIG_PCI_ENDPOINT_TEST)	+= pci_endpoint_test.o | ||||
| obj-$(CONFIG_OCXL)		+= ocxl/ | ||||
| obj-$(CONFIG_BCM_VK)		+= bcm-vk/ | ||||
|  | ||||
							
								
								
									
										420
									
								
								drivers/misc/dw-xdata-pcie.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										420
									
								
								drivers/misc/dw-xdata-pcie.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,420 @@ | ||||
| // 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_simple_get(&xdata_ida, 0, 0, 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_simple_remove(&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_simple_remove(&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>"); | ||||
| 
 | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user