Received: by 2002:a05:6a10:8c0a:0:0:0:0 with SMTP id go10csp679333pxb; Tue, 2 Feb 2021 15:19:14 -0800 (PST) X-Google-Smtp-Source: ABdhPJyoJxsQ2UR7PRFv3uTQRSG2J8PfzaLO8fJCMc1n42VrODqVbqcIaFi9x1/ymAhBRJZjI9OO X-Received: by 2002:a05:6402:34c3:: with SMTP id w3mr419553edc.3.1612307953999; Tue, 02 Feb 2021 15:19:13 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1612307953; cv=none; d=google.com; s=arc-20160816; b=AyAHRh4S3Dy2jDI4xjBeu7yj9nIi9nJeHAR78Ln8SdXY0K2CkvnL6gz6Hb4WMMJOtn vJjSgpPBRj5irexwwc5/AoMP5mzxoQqSyxWNUxhegb/XTaYq+fpitz9aZegG60Mf4e+3 zd0p5sW0EhMmi+sacMeGOqAW9GIncN64LtpHBXEpSOvtOXnKDQhfd/s/EeH7SwCxAjBu CRVuyQx7kx13T4nvKSo9GjaW1rOkUaacpZVUeB1PxMiEf/nVemnw3bTzlKqit6ERBnJh lVW6CsXSIvku1+vwikEiNgO5TI/eUd62lGelMPdHA0DR9Z9xlx2XZzxowETgThcnMWGf Ymrw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:references:in-reply-to:references:in-reply-to :message-id:date:subject:to:from:dkim-signature; bh=5fC+2GlGIf+57q80VeqAmm+vLyEImWp5g6XpH+UNzLY=; b=OA+AEOOn7CpB3VLenhUfkURSpjKtE9NwcenhRa2sswVSCgAChyx55SIigAqg4mMotL sw/yL+naUDPO2aZ8DigEBqLydNf+QQ+iA/Vpm0GqiyJl7FhKCOtQLeMz4bkV8i5dD99i 1v2YIhy/90Cx1oddSHy67ObT0KNpwO5Z/doCG6lRIIrYFZRiMOdnZCX41IcNX7ZA3rnP eHhgRVKoncUVfcNMkprriEF4si3293N/UpdMnYWs4/7tsPupLA5pwqUPv258re1MKBDC Q5DcgHG1M6Se8kQDqjVszn0qniE+v11PGahRG0vOyG/C67H0yY7+DWS4MYo9zwsoTiKz 2KWg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@synopsys.com header.s=mail header.b=U2I6hLA4; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=synopsys.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id bs11si178528ejb.637.2021.02.02.15.18.49; Tue, 02 Feb 2021 15:19:13 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@synopsys.com header.s=mail header.b=U2I6hLA4; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=synopsys.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S236937AbhBBRAD (ORCPT + 99 others); Tue, 2 Feb 2021 12:00:03 -0500 Received: from smtprelay-out1.synopsys.com ([149.117.73.133]:53106 "EHLO smtprelay-out1.synopsys.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S237024AbhBBQ5v (ORCPT ); Tue, 2 Feb 2021 11:57:51 -0500 Received: from mailhost.synopsys.com (mdc-mailhost2.synopsys.com [10.225.0.210]) (using TLSv1.3 with cipher TLS_AES_256_GCM_SHA384 (256/256 bits)) (No client certificate requested) by smtprelay-out1.synopsys.com (Postfix) with ESMTPS id C3AFD401D7; Tue, 2 Feb 2021 16:56:46 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=synopsys.com; s=mail; t=1612285007; bh=hjAVmHC62hqnPFQCk0S7Hm21hPzEXOtAr+1vKzySnq0=; h=From:To:Subject:Date:In-Reply-To:References:In-Reply-To: References:From; b=U2I6hLA43fj4uKzLnwQFsLXEHhWaoRNGz2MFru5zEEafLoIjO70oemUVwP0BY5lQg vcYoxj/FZf+i0RdCGbdRB0Ygge2wdq0CH4SdkmOrKgTuYBib8bG1nEMQyIcLhgZoFY u38IrNJ2UqcRIVu/rSFWDBm41IjZJ4O+3VYu1cEvT2zWVPTAHSPShr1SDQKlOzz3dS o6PDk/L7g8LQUmZiuKzs+IJ58o+GtDGHjz0X6MU4CpZKmLmjfPj8WZnaLyk5EcOaL/ aNcy1pU6RE+kJK/eY3Pe1gGrGEiGQCqY3pAqcsC/mWYytqP+qmztNPrN3rHcActpDx ken9GLZVcScuw== Received: from de02dwia024.internal.synopsys.com (de02dwia024.internal.synopsys.com [10.225.19.81]) by mailhost.synopsys.com (Postfix) with ESMTP id 8F610A0062; Tue, 2 Feb 2021 16:56:45 +0000 (UTC) X-SNPS-Relay: synopsys.com From: Gustavo Pimentel To: linux-pci@vger.kernel.org, linux-kernel@vger.kernel.org, dmaengine@vger.kernel.org, Derek Kiernan , Dragan Cvetic , Arnd Bergmann , Andrew Morton , Greg Kroah-Hartman , Jonathan Corbet , Gustavo Pimentel Subject: [RESEND PATCH v3 1/5] misc: Add Synopsys DesignWare xData IP driver Date: Tue, 2 Feb 2021 17:56:34 +0100 Message-Id: <2c70018d5965c171c15870638ee717fe5f9483f6.1612284945.git.gustavo.pimentel@synopsys.com> X-Mailer: git-send-email 2.7.4 In-Reply-To: References: In-Reply-To: References: Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org 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 --- drivers/misc/dw-xdata-pcie.c | 379 +++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 379 insertions(+) create mode 100644 drivers/misc/dw-xdata-pcie.c diff --git a/drivers/misc/dw-xdata-pcie.c b/drivers/misc/dw-xdata-pcie.c new file mode 100644 index 00000000..683bfda --- /dev/null +++ b/drivers/misc/dw-xdata-pcie.c @@ -0,0 +1,379 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * Copyright (c) 2020 Synopsys, Inc. and/or its affiliates. + * Synopsys DesignWare xData driver + * + * Author: Gustavo Pimentel + */ + +#include +#include +#include +#include +#include +#include +#include +#include + +#define DW_XDATA_DRIVER_NAME "dw-xdata-pcie" + +#define DW_XDATA_EP_MEM_OFFSET 0x8000000 + +struct dw_xdata_pcie_data { + /* xData registers location */ + enum pci_barno rg_bar; + off_t rg_off; + size_t rg_sz; +}; + +static const struct dw_xdata_pcie_data snps_edda_data = { + /* xData registers location */ + .rg_bar = BAR_0, + .rg_off = 0x00000000, /* 0 Kbytes */ + .rg_sz = 0x0000012c, /* 300 bytes */ +}; + +#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) + +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 */ + size_t sz; /* size */ +}; + +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 pci_dev *pdev; +}; + +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 = readl(&(__dw_regs(dw)->burst_cnt)); + + if (burst & BIT(31)) { + burst &= ~(u32)BIT(31); + writel(burst, &(__dw_regs(dw)->burst_cnt)); + } +} + +static void dw_xdata_start(struct dw_xdata *dw, bool write) +{ + u32 control, status; + + /* Stop first if xfer in progress */ + dw_xdata_stop(dw); + + /* Clear status register */ + writel(0x0, &(__dw_regs(dw)->status)); + + /* Burst count register set for continuous until stopped */ + writel(0x80001001, &(__dw_regs(dw)->burst_cnt)); + + /* Pattern register */ + writel(0x0, &(__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)); + + usleep_range(100, 150); + + status = readl(&(__dw_regs(dw)->status)); + if (!(status & STATUS_DONE)) + pci_dbg(dw->pdev, "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) +{ + u64 data[2], time[2], diff; + + /* First measurement */ + 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)); + + /* Delay 100ms */ + mdelay(100); + + /* Second measurement */ + 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)); + + /* Calculations */ + diff = jiffies_to_nsecs(time[1] - time[0]); + *rate = dw_xdata_perf_diff(&data[1], &data[0], diff); + + pci_dbg(dw->pdev, "xData: time=%llu us, %s=%llu MB/s\n", + diff, write ? "write" : "read", *rate); +} + +static inline struct device *kobj2device(struct kobject *kobj) +{ + return container_of(kobj, struct device, kobj); +} + +static inline struct pci_dev *device2pci_dev(struct device *dev) +{ + return container_of(dev, struct pci_dev, dev); +} + +static ssize_t sysfs_write_show(struct kobject *kobj, + struct kobj_attribute *attr, char *buf) +{ + struct device *dev = kobj2device(kobj); + struct pci_dev *pdev = device2pci_dev(dev); + struct dw_xdata *dw = pci_get_drvdata(pdev); + u64 rate; + + dw_xdata_perf(dw, &rate, true); + return sprintf(buf, "%llu MB/s\n", rate); +} + +static ssize_t sysfs_write_store(struct kobject *kobj, + struct kobj_attribute *attr, const char *buf, + size_t count) +{ + struct device *dev = kobj2device(kobj); + struct pci_dev *pdev = device2pci_dev(dev); + struct dw_xdata *dw = pci_get_drvdata(pdev); + + pci_dbg(pdev, "xData: requested write transfer\n"); + dw_xdata_start(dw, true); + + return count; +} + +struct kobj_attribute sysfs_write_attr = __ATTR(write, 0644, + sysfs_write_show, + sysfs_write_store); + +static ssize_t sysfs_read_show(struct kobject *kobj, + struct kobj_attribute *attr, char *buf) +{ + struct device *dev = kobj2device(kobj); + struct pci_dev *pdev = device2pci_dev(dev); + struct dw_xdata *dw = pci_get_drvdata(pdev); + u64 rate; + + dw_xdata_perf(dw, &rate, false); + return sprintf(buf, "%llu MB/s\n", rate); +} + +static ssize_t sysfs_read_store(struct kobject *kobj, + struct kobj_attribute *attr, const char *buf, + size_t count) +{ + struct device *dev = kobj2device(kobj); + struct pci_dev *pdev = device2pci_dev(dev); + struct dw_xdata *dw = pci_get_drvdata(pdev); + + pci_dbg(pdev, "xData: requested read transfer\n"); + dw_xdata_start(dw, false); + + return count; +} + +struct kobj_attribute sysfs_read_attr = __ATTR(read, 0644, + sysfs_read_show, + sysfs_read_store); + +static ssize_t sysfs_stop_store(struct kobject *kobj, + struct kobj_attribute *attr, const char *buf, + size_t count) +{ + struct device *dev = kobj2device(kobj); + struct pci_dev *pdev = device2pci_dev(dev); + struct dw_xdata *dw = pci_get_drvdata(pdev); + + pci_dbg(pdev, "xData: requested stop any transfer\n"); + dw_xdata_stop(dw); + + return count; +} + +struct kobj_attribute sysfs_stop_attr = __ATTR(stop, 0644, + NULL, + sysfs_stop_store); + +static int dw_xdata_pcie_probe(struct pci_dev *pdev, + const struct pci_device_id *pid) +{ + const struct dw_xdata_pcie_data *pdata = (void *)pid->driver_data; + struct device *dev = &pdev->dev; + struct dw_xdata *dw; + u64 addr; + int err; + + /* Enable PCI device */ + err = pcim_enable_device(pdev); + if (err) { + pci_err(pdev, "enabling device failed\n"); + return err; + } + + /* Mapping PCI BAR regions */ + err = pcim_iomap_regions(pdev, BIT(pdata->rg_bar), pci_name(pdev)); + if (err) { + pci_err(pdev, "xData BAR I/O remapping failed\n"); + return err; + } + + pci_set_master(pdev); + + /* Allocate memory */ + dw = devm_kzalloc(&pdev->dev, sizeof(*dw), GFP_KERNEL); + if (!dw) + return -ENOMEM; + + /* Data structure initialization */ + dw->rg_region.vaddr = pcim_iomap_table(pdev)[pdata->rg_bar]; + if (!dw->rg_region.vaddr) + return -ENOMEM; + + dw->rg_region.vaddr += pdata->rg_off; + dw->rg_region.paddr = pdev->resource[pdata->rg_bar].start; + dw->rg_region.paddr += pdata->rg_off; + dw->rg_region.sz = pdata->rg_sz; + + 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; + + 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)); + pci_dbg(pdev, "xData: target address = 0x%.16llx\n", addr); + + pci_dbg(pdev, "xData: wr_len=%zu, rd_len=%zu\n", + dw->max_wr_len * 4, dw->max_rd_len * 4); + + err = sysfs_create_file(&dev->kobj, &sysfs_write_attr.attr); + if (err) + return err; + + err = sysfs_create_file(&dev->kobj, &sysfs_read_attr.attr); + if (err) + return err; + + err = sysfs_create_file(&dev->kobj, &sysfs_stop_attr.attr); + if (err) + return err; + + err = sysfs_create_link(kernel_kobj, &dev->kobj, DW_XDATA_DRIVER_NAME); + if (err) + return err; + + /* Saving data structure reference */ + pci_set_drvdata(pdev, dw); + + return 0; +} + +static void dw_xdata_pcie_remove(struct pci_dev *pdev) +{ + struct dw_xdata *dw = pci_get_drvdata(pdev); + struct device *dev = &pdev->dev; + + if (dw) + dw_xdata_stop(dw); + + sysfs_remove_link(kernel_kobj, DW_XDATA_DRIVER_NAME); + sysfs_remove_file(&dev->kobj, &sysfs_stop_attr.attr); + sysfs_remove_file(&dev->kobj, &sysfs_read_attr.attr); + sysfs_remove_file(&dev->kobj, &sysfs_write_attr.attr); + kobject_put(&dev->kobj); +} + +static const struct pci_device_id dw_xdata_pcie_id_table[] = { + { PCI_DEVICE_DATA(SYNOPSYS, EDDA, &snps_edda_data) }, + { } +}; +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 "); + -- 2.7.4