From patchwork Wed Jul 8 16:11:25 2015 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Peter Griffin X-Patchwork-Id: 50898 Return-Path: X-Original-To: linaro@patches.linaro.org Delivered-To: linaro@patches.linaro.org Received: from mail-lb0-f197.google.com (mail-lb0-f197.google.com [209.85.217.197]) by ip-10-151-82-157.ec2.internal (Postfix) with ESMTPS id 5DB7E22A03 for ; Wed, 8 Jul 2015 16:13:44 +0000 (UTC) Received: by lbcpe5 with SMTP id pe5sf25463545lbc.3 for ; Wed, 08 Jul 2015 09:13:43 -0700 (PDT) X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20130820; h=x-gm-message-state:delivered-to:from:to:subject:date:message-id :in-reply-to:references:precedence:list-id:list-unsubscribe :list-archive:list-post:list-help:list-subscribe:cc:mime-version :content-type:content-transfer-encoding:sender:errors-to :x-original-sender:x-original-authentication-results:mailing-list; bh=QRdwTgCWcUD5oeyzFsv1w2Z2A+mUMh2A9+ajVswEIVs=; b=mHyi2tFObN6VmFmGeAx58mMzwzFdDcp4Tu7Ee3d9QmwBtNblCcFm5dqxduGotTf63+ 9y49abDnNn4N249laMUjb43YUiaAcFEibWiDuJzgyUTMoc6r/OmBJW/FoROHP8owYK1Z xjXu1VJCIaGrmEVqNqUwqY8X3CWjv0hvDdhy9ngtaL/Jt/HqU2ZMSDfLAOeIcocDnh1/ zD9LsiVu4Gn26SrevzVwEMS/tKo0mwUbE3n/0i4bXYXIfY65lbVc/sKp3HT5iFvMY6zb gr+hIJuJa/w08Qjdll2dkjO3ZEdFy3tCS0yomOfgAHw8Zg73EsHhgs71XnnU4VugqynG clnw== X-Gm-Message-State: ALoCoQkbHGad7PP5YJ4RmN+hc5oY6UQq0rMl8f36BM9lt1BiYFfTe0Lwk3VKMYjWZ8ibsVZWp4+Q X-Received: by 10.180.105.226 with SMTP id gp2mr34103439wib.1.1436372023356; Wed, 08 Jul 2015 09:13:43 -0700 (PDT) X-BeenThere: patchwork-forward@linaro.org Received: by 10.152.8.201 with SMTP id t9ls918571laa.95.gmail; Wed, 08 Jul 2015 09:13:43 -0700 (PDT) X-Received: by 10.152.1.66 with SMTP id 2mr8695554lak.56.1436372023188; Wed, 08 Jul 2015 09:13:43 -0700 (PDT) Received: from mail-la0-f47.google.com (mail-la0-f47.google.com. [209.85.215.47]) by mx.google.com with ESMTPS id mi10si2331664lbb.154.2015.07.08.09.13.43 for (version=TLSv1.2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Wed, 08 Jul 2015 09:13:43 -0700 (PDT) Received-SPF: pass (google.com: domain of patch+caf_=patchwork-forward=linaro.org@linaro.org designates 209.85.215.47 as permitted sender) client-ip=209.85.215.47; Received: by lagc2 with SMTP id c2so225879099lag.3 for ; Wed, 08 Jul 2015 09:13:43 -0700 (PDT) X-Received: by 10.152.18.162 with SMTP id x2mr10344890lad.73.1436372023024; Wed, 08 Jul 2015 09:13:43 -0700 (PDT) X-Forwarded-To: patchwork-forward@linaro.org X-Forwarded-For: patch@linaro.org patchwork-forward@linaro.org Delivered-To: patch@linaro.org Received: by 10.112.108.230 with SMTP id hn6csp61309lbb; Wed, 8 Jul 2015 09:13:41 -0700 (PDT) X-Received: by 10.70.42.101 with SMTP id n5mr22301748pdl.93.1436372016305; Wed, 08 Jul 2015 09:13:36 -0700 (PDT) Received: from bombadil.infradead.org (bombadil.infradead.org. [2001:1868:205::9]) by mx.google.com with ESMTPS id ez4si4866439pbb.16.2015.07.08.09.13.35 for (version=TLSv1.2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Wed, 08 Jul 2015 09:13:36 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-arm-kernel-bounces+patch=linaro.org@lists.infradead.org designates 2001:1868:205::9 as permitted sender) client-ip=2001:1868:205::9; Received: from localhost ([127.0.0.1] helo=bombadil.infradead.org) by bombadil.infradead.org with esmtp (Exim 4.80.1 #2 (Red Hat Linux)) id 1ZCrxP-0000OU-Fr; Wed, 08 Jul 2015 16:12:23 +0000 Received: from mail-wi0-f172.google.com ([209.85.212.172]) by bombadil.infradead.org with esmtps (Exim 4.80.1 #2 (Red Hat Linux)) id 1ZCrx6-0008Fo-Dj for linux-arm-kernel@lists.infradead.org; Wed, 08 Jul 2015 16:12:05 +0000 Received: by wibdq8 with SMTP id dq8so217598629wib.1 for ; Wed, 08 Jul 2015 09:11:42 -0700 (PDT) X-Received: by 10.194.110.100 with SMTP id hz4mr21342218wjb.6.1436371902750; Wed, 08 Jul 2015 09:11:42 -0700 (PDT) Received: from localhost.localdomain (cpc14-aztw22-2-0-cust189.18-1.cable.virginm.net. [82.45.1.190]) by smtp.gmail.com with ESMTPSA id c3sm4230830wja.3.2015.07.08.09.11.41 (version=TLSv1.2 cipher=ECDHE-RSA-AES128-SHA bits=128/128); Wed, 08 Jul 2015 09:11:41 -0700 (PDT) From: Peter Griffin To: linux-arm-kernel@lists.infradead.org, linux-kernel@vger.kernel.org, srinivas.kandagatla@gmail.com, maxime.coquelin@st.com, patrice.chotard@st.com, vinod.koul@intel.com, dan.j.williams@intel.com Subject: [PATCH 4/7] dmaengine: st_fdma: Add xbar support Date: Wed, 8 Jul 2015 17:11:25 +0100 Message-Id: <1436371888-27863-5-git-send-email-peter.griffin@linaro.org> X-Mailer: git-send-email 1.9.1 In-Reply-To: <1436371888-27863-1-git-send-email-peter.griffin@linaro.org> References: <1436371888-27863-1-git-send-email-peter.griffin@linaro.org> X-CRM114-Version: 20100106-BlameMichelson ( TRE 0.8.0 (BSD) ) MR-646709E3 X-CRM114-CacheID: sfid-20150708_091204_775273_B24FF679 X-CRM114-Status: GOOD ( 23.98 ) X-Spam-Score: -2.6 (--) X-Spam-Report: SpamAssassin version 3.4.0 on bombadil.infradead.org summary: Content analysis details: (-2.6 points) pts rule name description ---- ---------------------- -------------------------------------------------- -0.7 RCVD_IN_DNSWL_LOW RBL: Sender listed at http://www.dnswl.org/, low trust [209.85.212.172 listed in list.dnswl.org] -0.0 RCVD_IN_MSPIKE_H3 RBL: Good reputation (+3) [209.85.212.172 listed in wl.mailspike.net] -0.0 SPF_PASS SPF: sender matches SPF record -1.9 BAYES_00 BODY: Bayes spam probability is 0 to 1% [score: 0.0000] -0.0 RCVD_IN_MSPIKE_WL Mailspike good senders X-BeenThere: linux-arm-kernel@lists.infradead.org X-Mailman-Version: 2.1.20 Precedence: list List-Id: List-Unsubscribe: , List-Archive: List-Post: , List-Help: , List-Subscribe: , Cc: peter.griffin@linaro.org, devicetree@vger.kernel.org, lee.jones@linaro.org, dmaengine@vger.kernel.org, ludovic.barre@st.com MIME-Version: 1.0 Sender: "linux-arm-kernel" Errors-To: linux-arm-kernel-bounces+patch=linaro.org@lists.infradead.org X-Removed-Original-Auth: Dkim didn't pass. X-Original-Sender: peter.griffin@linaro.org X-Original-Authentication-Results: mx.google.com; spf=pass (google.com: domain of patch+caf_=patchwork-forward=linaro.org@linaro.org designates 209.85.215.47 as permitted sender) smtp.mail=patch+caf_=patchwork-forward=linaro.org@linaro.org Mailing-list: list patchwork-forward@linaro.org; contact patchwork-forward+owners@linaro.org X-Google-Group-Id: 836684582541 To increase the number of peripheral requests, the FDMA crossbar can multiplex up to 96 peripheral requests to one of 3 fdma controllers. Signed-off-by: Ludovic Barre Signed-off-by: Peter Griffin --- drivers/dma/Kconfig | 11 ++++ drivers/dma/Makefile | 1 + drivers/dma/st_fdma.c | 25 +++++++- drivers/dma/st_fdma.h | 32 ++++++++++ drivers/dma/st_fdma_xbar.c | 149 +++++++++++++++++++++++++++++++++++++++++++++ 5 files changed, 216 insertions(+), 2 deletions(-) create mode 100644 drivers/dma/st_fdma_xbar.c diff --git a/drivers/dma/Kconfig b/drivers/dma/Kconfig index 7a016e0..21802b1 100644 --- a/drivers/dma/Kconfig +++ b/drivers/dma/Kconfig @@ -522,4 +522,15 @@ config ST_FDMA Say Y here if you have such a chipset. If unsure, say N. +config ST_FDMA_XBAR + bool "ST FDMA crossbar" + depends on ST_FDMA + default y + help + Enable support for ST FDMA crossbar. + xbar add flexibility and increase the number of peripheral request + can be used by fdma xbar can multiplex until 96 peripheral requests + to one of 3 fdma controller + + endif diff --git a/drivers/dma/Makefile b/drivers/dma/Makefile index f68e6d8..19f18b1 100644 --- a/drivers/dma/Makefile +++ b/drivers/dma/Makefile @@ -50,6 +50,7 @@ obj-$(CONFIG_MOXART_DMA) += moxart-dma.o obj-$(CONFIG_FSL_RAID) += fsl_raid.o obj-$(CONFIG_FSL_EDMA) += fsl-edma.o obj-$(CONFIG_QCOM_BAM_DMA) += qcom_bam_dma.o +obj-$(CONFIG_ST_FDMA_XBAR) += st_fdma_xbar.o obj-$(CONFIG_ST_FDMA) += st_fdma.o obj-y += xilinx/ obj-$(CONFIG_INTEL_MIC_X100_DMA) += mic_x100_dma.o diff --git a/drivers/dma/st_fdma.c b/drivers/dma/st_fdma.c index 07a6df1..a870902 100644 --- a/drivers/dma/st_fdma.c +++ b/drivers/dma/st_fdma.c @@ -336,7 +336,9 @@ static int st_fdma_dreq_get(struct st_fdma_chan *fchan) return -EINVAL; } - if (try || req_line_cfg >= ST_FDMA_NR_DREQS) { + if (fdev->xbar_dev) + dreq_line = ffz(fdev->dreq_mask); + else if (try || req_line_cfg >= ST_FDMA_NR_DREQS) { dev_err(fdev->dev, "Invalid or used req line\n"); return -EINVAL; } else { @@ -903,6 +905,15 @@ static int st_fdma_slave_config(struct dma_chan *chan, else return -EINVAL; + if (fdev->xbar_dev) { + if (st_fdma_xbar_mux(fdev->xbar_dev, fchan->cfg.req_line, + fchan->dreq_line)) { + dev_err(fdev->dev, "Error routing req line\n"); + clear_bit(fchan->dreq_line, &fdev->dreq_mask); + return -EINVAL; + } + } + fchan->cfg.req_ctrl |= REQ_CTRL_NUM_OPS(maxburst-1); dreq_write(fchan, fchan->cfg.req_ctrl, REQ_CTRL); @@ -1044,6 +1055,13 @@ static int st_fdma_probe(struct platform_device *pdev) if (ret) goto err_clk; + fdev->xbar_dev = st_fdma_xbar_request(&pdev->dev); + if (IS_ERR(fdev->xbar_dev)) { + dev_err(&pdev->dev, "Failed to request xbar:%ld\n", + PTR_ERR(fdev->xbar_dev)); + goto err_clk; + } + /* Initialise the FDMA dreq (reserve 0 & 31 for FDMA use) */ fdev->dreq_mask = BIT(0) | BIT(31); @@ -1081,7 +1099,10 @@ static int st_fdma_probe(struct platform_device *pdev) goto err_dma_dev; } - dev_info(&pdev->dev, "ST FDMA engine driver, irq:%d\n", irq); + dev_info(&pdev->dev, "ST FDMA engine driver, irq:%d xbar(%s, id:%d)\n", + irq, fdev->xbar_dev ? + dev_name(&fdev->xbar_dev->pdev->dev) : "no", + fdev->xbar_dev ? fdev->xbar_dev->fdma_id : 0); return 0; diff --git a/drivers/dma/st_fdma.h b/drivers/dma/st_fdma.h index 533c811..dd46780 100644 --- a/drivers/dma/st_fdma.h +++ b/drivers/dma/st_fdma.h @@ -17,6 +17,20 @@ #define ST_FDMA_NR_DREQS 32 #define EM_SLIM 102 /* No official SLIM ELF ID */ +struct st_fdma_xbar { + void __iomem *io_base; + struct clk *clk; + u32 nr_requests; + u32 max_nr_requests; +}; + +struct st_fdma_xbar_dev { + struct platform_device *pdev; + struct st_fdma_xbar *xbar; + struct device *fdev; + u32 fdma_id; +}; + enum { CLK_SLIM, CLK_HI, @@ -135,6 +149,7 @@ struct st_fdma_dev { struct st_fdma_chan *chans; + struct st_fdma_xbar_dev *xbar_dev; spinlock_t dreq_lock; unsigned long dreq_mask; @@ -142,6 +157,23 @@ struct st_fdma_dev { atomic_t fw_loaded; }; +#if defined(CONFIG_ST_FDMA_XBAR) +struct st_fdma_xbar_dev *st_fdma_xbar_request(struct device *device); +int st_fdma_xbar_mux(struct st_fdma_xbar_dev *xbar_dev, + u32 in_dreq, u32 fdma_dreq); +#else +struct st_fdma_xbar_dev *st_fdma_xbar_request(struct device *device) +{ + return NULL; +} + +int st_fdma_xbar_mux(struct st_fdma_xbar_dev *xbar_dev, + u32 in_dreq, u32 fdma_dreq) +{ + return -ENODEV; +} +#endif + /* Registers*/ /* FDMA interface */ #define FDMA_ID_OFST 0x00000 diff --git a/drivers/dma/st_fdma_xbar.c b/drivers/dma/st_fdma_xbar.c new file mode 100644 index 0000000..c3b8e0f --- /dev/null +++ b/drivers/dma/st_fdma_xbar.c @@ -0,0 +1,149 @@ +/* + * st_fdma_xbar.c + * + * Copyright (C) 2014 STMicroelectronics + * Author: Ludovic Barre + * License terms: GNU General Public License (GPL), version 2 + */ +#include +#include +#include +#include +#include +#include + +#include "st_fdma.h" + +#define XBAR_1_0_MAX_REQ 96 + +int st_fdma_xbar_mux(struct st_fdma_xbar_dev *xbar_dev, + u32 per_req, u32 fdma_req) +{ + u32 out_req; + + if (per_req >= xbar_dev->xbar->nr_requests) + return -EINVAL; + + out_req = (xbar_dev->fdma_id * ST_FDMA_NR_DREQS) + fdma_req; + writel(per_req, xbar_dev->xbar->io_base + (out_req << 2)); + + dev_dbg(&xbar_dev->pdev->dev, "in_req:%d out_req:%d\n", + per_req, out_req); + + return 0; +} + +struct st_fdma_xbar_dev *st_fdma_xbar_request(struct device *device) +{ + struct device_node *np = device->of_node; + struct st_fdma_xbar_dev *xbar_dev; + struct of_phandle_args args; + int err; + + err = of_parse_phandle_with_args(np, "st,fdma-xbar", + "#st,fdma-xbar-cells", 0, &args); + if (err) + return NULL; + + xbar_dev = kzalloc(sizeof(*xbar_dev), GFP_KERNEL); + if (!xbar_dev) { + of_node_put(args.np); + err = -ENOMEM; + goto out; + } + + xbar_dev->pdev = of_find_device_by_node(args.np); + if (!xbar_dev->pdev) { + of_node_put(args.np); + err = -ENODEV; + goto free; + } + + of_node_put(args.np); + + xbar_dev->xbar = platform_get_drvdata(xbar_dev->pdev); + if (!xbar_dev->xbar) { + err = -EPROBE_DEFER; + goto pdev_put; + } + + xbar_dev->fdma_id = args.args[0]; + xbar_dev->fdev = device; + + return xbar_dev; + +pdev_put: + platform_device_put(xbar_dev->pdev); +free: + kfree(xbar_dev); +out: + return ERR_PTR(err); +} + +void st_fdma_xbar_free(struct st_fdma_xbar_dev *device) +{ + platform_device_put(device->pdev); + kfree(device); +} + +static const struct of_device_id st_fdma_xbar_match[] = { + { .compatible = "st,fdma-xbar-1.0", .data = (void *)XBAR_1_0_MAX_REQ }, + {}, +}; +MODULE_DEVICE_TABLE(of, st_fdma_xbar_match); + +static int st_fdma_xbar_probe(struct platform_device *pdev) +{ + const struct of_device_id *match; + struct device_node *np = pdev->dev.of_node; + struct st_fdma_xbar *xbar; + struct resource *res; + + match = of_match_device((st_fdma_xbar_match), &pdev->dev); + if (!match) { + dev_err(&pdev->dev, "No device match found\n"); + return -ENODEV; + } + + xbar = devm_kzalloc(&pdev->dev, sizeof(*xbar), GFP_KERNEL); + if (!xbar) + return -ENOMEM; + + res = platform_get_resource(pdev, IORESOURCE_MEM, 0); + xbar->io_base = devm_ioremap_resource(&pdev->dev, res); + if (IS_ERR(xbar->io_base)) + return PTR_ERR(xbar->io_base); + + xbar->max_nr_requests = (unsigned long)match->data; + + if (of_property_read_u32(np, "dma-requests", &xbar->nr_requests)) + xbar->nr_requests = xbar->max_nr_requests; + + if (xbar->nr_requests > xbar->max_nr_requests) { + dev_err(&pdev->dev, "Nr requests not supported\n"); + return -EINVAL; + } + + platform_set_drvdata(pdev, xbar); + + return 0; +} + +static int st_fdma_xbar_remove(struct platform_device *pdev) +{ + return 0; +} + +static struct platform_driver st_fdma_xbar_driver = { + .driver = { + .name = "st-fdma-xbar", + .of_match_table = st_fdma_xbar_match, + }, + .probe = st_fdma_xbar_probe, + .remove = st_fdma_xbar_remove, +}; +module_platform_driver(st_fdma_xbar_driver); + +MODULE_LICENSE("GPL v2"); +MODULE_DESCRIPTION("STMicroelectronics FDMA cross bar"); +MODULE_AUTHOR("Ludovic.barre ");