Received: by 2002:ac0:a5a6:0:0:0:0:0 with SMTP id m35-v6csp2016856imm; Sat, 29 Sep 2018 08:40:50 -0700 (PDT) X-Google-Smtp-Source: ACcGV60XL44Z+nIwTDDKhRbJag98uwwL9T0R4s8MhKRCziANk9gnkViokQgbx50VfLoEEEo7cdjH X-Received: by 2002:a63:f314:: with SMTP id l20-v6mr371079pgh.407.1538235650712; Sat, 29 Sep 2018 08:40:50 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1538235650; cv=none; d=google.com; s=arc-20160816; b=edwjjPUqOvsS6IiRaSqxf4FLAGvHKWdqxgZss6Deaqk9mheNHgq3eDhRW9qkekLALJ wtRsm+wO+y7RpccuQsIBDosuBr5LEnk0MvSFzgCRbVfj3SFpB3KB6dRhp/Woh7n80qRl Q/OTt3o2xqAT/0NWIuH1o1//sn4YFBlYVgswlpL8aeglMeJAIloBaF7e39aXlo1cjc+O HjmcqJIxjY9ll+lq+vC2hgyg/vh/aFENBNnL8SpoUl4Ei2nX0+1nYevZOTJNK4W0qRPg gw3T885yYlJyNUN5dvlAw/Ii+yxim/D5DK+y2e5HpwOIGHt3ZHvW8guaJ4/UwKePOLfe kzZA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :references:in-reply-to:message-id:subject:cc:to:from:date; bh=vK+gAqo/N3zaQvVvawmiVQH3SDHq2CQJcsgBkoUxVxQ=; b=xKQ5RhbD9xAFZwdNc/GWlW57Wbs2+l+591yDvVSGioRUt+xbzCWE1DhmJ7/EWEej8N GTK/95QSsfmCMMLfiwEINA/JQOTyjUCCCczTN9HABl4EShNvr0csDuxVcc0vL1TKlAYd c43c/WR5stZZA0tdcvTB2eUjM/V9GFpKrsVWIAG5zDCeTc5j5Qo8YcOcrnA7BpqJqMtH +1IofpyhVcIGZoo27lsUOYS/U4st/sw3XLh1XGqICVFT1O73lGlc7CsDOOfpzYcjSFw3 vUn/sY32PJU8NrmQB/aV9u52vfgTQEBsBbwPKsyAKq11Mu3hLTFkpaOvJn6l1cqBLnVI VLCQ== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id t10-v6si4766351plh.474.2018.09.29.08.40.35; Sat, 29 Sep 2018 08:40:50 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1728352AbeI2WJX (ORCPT + 99 others); Sat, 29 Sep 2018 18:09:23 -0400 Received: from mail.bootlin.com ([62.4.15.54]:55972 "EHLO mail.bootlin.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1728123AbeI2WJX (ORCPT ); Sat, 29 Sep 2018 18:09:23 -0400 Received: by mail.bootlin.com (Postfix, from userid 110) id 4646220734; Sat, 29 Sep 2018 17:40:27 +0200 (CEST) X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on mail.bootlin.com X-Spam-Level: X-Spam-Status: No, score=-1.0 required=5.0 tests=ALL_TRUSTED,SHORTCIRCUIT shortcircuit=ham autolearn=disabled version=3.4.0 Received: from bbrezillon (unknown [91.160.177.164]) by mail.bootlin.com (Postfix) with ESMTPSA id DAACB20711; Sat, 29 Sep 2018 17:40:26 +0200 (CEST) Date: Sat, 29 Sep 2018 17:40:23 +0200 From: Boris Brezillon To: Yogesh Gaur Cc: linux-mtd@lists.infradead.org, marek.vasut@gmail.com, linux-spi@vger.kernel.org, devicetree@vger.kernel.org, robh@kernel.org, mark.rutland@arm.com, shawnguo@kernel.org, linux-arm-kernel@lists.infradead.org, computersforpeace@gmail.com, frieder.schrempf@exceet.de, linux-kernel@vger.kernel.org Subject: Re: [PATCH v3 1/5] spi: spi-mem: Add driver for NXP FlexSPI controller Message-ID: <20180929174023.51b1e284@bbrezillon> In-Reply-To: <1537525323-20730-2-git-send-email-yogeshnarayan.gaur@nxp.com> References: <1537525323-20730-1-git-send-email-yogeshnarayan.gaur@nxp.com> <1537525323-20730-2-git-send-email-yogeshnarayan.gaur@nxp.com> X-Mailer: Claws Mail 3.15.0-dirty (GTK+ 2.24.31; x86_64-pc-linux-gnu) MIME-Version: 1.0 Content-Type: text/plain; charset=US-ASCII Content-Transfer-Encoding: 7bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Hi Yogesh, On Fri, 21 Sep 2018 15:51:59 +0530 Yogesh Gaur wrote: > +/* Registers used by the driver */ > +#define FSPI_MCR0 0x00 > +#define FSPI_MCR0_AHB_TIMEOUT_SHIFT 24 > +#define FSPI_MCR0_AHB_TIMEOUT_MASK (0xFF << FSPI_MCR0_AHB_TIMEOUT_SHIFT) > +#define FSPI_MCR0_IP_TIMEOUT_SHIFT 16 > +#define FSPI_MCR0_IP_TIMEOUT_MASK (0xFF << FSPI_MCR0_IP_TIMEOUT_SHIFT) > +#define FSPI_MCR0_LEARN_EN_SHIFT 15 > +#define FSPI_MCR0_LEARN_EN_MASK (1 << FSPI_MCR0_LEARN_EN_SHIFT) > +#define FSPI_MCR0_SCRFRUN_EN_SHIFT 14 > +#define FSPI_MCR0_SCRFRUN_EN_MASK (1 << FSPI_MCR0_SCRFRUN_EN_SHIFT) > +#define FSPI_MCR0_OCTCOMB_EN_SHIFT 13 > +#define FSPI_MCR0_OCTCOMB_EN_MASK (1 << FSPI_MCR0_OCTCOMB_EN_SHIFT) > +#define FSPI_MCR0_DOZE_EN_SHIFT 12 > +#define FSPI_MCR0_DOZE_EN_MASK (1 << FSPI_MCR0_DOZE_EN_SHIFT) > +#define FSPI_MCR0_HSEN_SHIFT 11 > +#define FSPI_MCR0_HSEN_MASK (1 << FSPI_MCR0_HSEN_SHIFT) > +#define FSPI_MCR0_SERCLKDIV_SHIFT 8 > +#define FSPI_MCR0_SERCLKDIV_MASK (7 << FSPI_MCR0_SERCLKDIV_SHIFT) > +#define FSPI_MCR0_ATDF_EN_SHIFT 7 > +#define FSPI_MCR0_ATDF_EN_MASK (1 << FSPI_MCR0_ATDF_EN_SHIFT) > +#define FSPI_MCR0_ARDF_EN_SHIFT 6 > +#define FSPI_MCR0_ARDF_EN_MASK (1 << FSPI_MCR0_ARDF_EN_SHIFT) > +#define FSPI_MCR0_RXCLKSRC_SHIFT 4 > +#define FSPI_MCR0_RXCLKSRC_MASK (3 << FSPI_MCR0_RXCLKSRC_SHIFT) > +#define FSPI_MCR0_END_CFG_SHIFT 2 > +#define FSPI_MCR0_END_CFG_MASK (3 << FSPI_MCR0_END_CFG_SHIFT) > +#define FSPI_MCR0_MDIS_SHIFT 1 > +#define FSPI_MCR0_MDIS_MASK (1 << FSPI_MCR0_MDIS_SHIFT) > +#define FSPI_MCR0_SWRST_SHIFT 0 > +#define FSPI_MCR0_SWRST_MASK (1 << FSPI_MCR0_SWRST_SHIFT) Do we really need all those _SHIFT/_MASK defs? I mean #define FSPI_MCR0_SWRST BIT(0) or #define FSPI_MCR0_AHB_TIMEOUT(x) ((x) << 24) #define FSPI_MCR0_AHB_TIMEOUT_MASK GENMASK(31, 24) are just fine. > + > +enum nxp_fspi_devtype { > + NXP_FSPI_LX2160A, > +}; I'm pretty sure you don't need this enum if you describe all dev caps in the nxp_fspi_devtype_data struct. > + > +struct nxp_fspi_devtype_data { > + enum nxp_fspi_devtype devtype; > + unsigned int rxfifo; > + unsigned int txfifo; > + unsigned int ahb_buf_size; > + unsigned int quirks; > + bool endianness; How about renaming this variable big_endian and dropping the {L,B}_ENDIAN macros? > +}; [...] > +struct nxp_fspi { > + void __iomem *iobase; > + void __iomem *ahb_addr; > + u32 memmap_phy; > + u32 memmap_phy_size; > + struct clk *clk, *clk_en; > + struct device *dev; > + struct completion c; > + const struct nxp_fspi_devtype_data *devtype_data; > + struct mutex lock; > + struct pm_qos_request pm_qos_req; > + int selected; > + void (*write)(u32 val, void __iomem *addr); > + u32 (*read)(void __iomem *addr); > +}; > + > +static void fspi_writel_be(u32 val, void __iomem *addr) > +{ > + iowrite32be(val, addr); > +} > + > +static void fspi_writel(u32 val, void __iomem *addr) > +{ > + iowrite32(val, addr); > +} > + > +static u32 fspi_readl_be(void __iomem *addr) > +{ > + return ioread32be(addr); > +} > + > +static u32 fspi_readl(void __iomem *addr) > +{ > + return ioread32(addr); > +} Hm, I'd recommend dropping the ->read/write() hooks and providing the following functions: static void fspi_writel(struct nxp_fspi *f, u32 val, void __iomem *addr) { if (f->big_endian) iowrite32be(val, addr); else iowrite32(val, addr); } static u32 fspi_readl(struct nxp_fspi *f, void __iomem *addr) { if (f->big_endian) return ioread32be(addr); else return ioread32(addr); } > + > +static irqreturn_t nxp_fspi_irq_handler(int irq, void *dev_id) > +{ > + struct nxp_fspi *f = dev_id; > + u32 reg; > + > + /* clear interrupt */ > + reg = f->read(f->iobase + FSPI_INTR); > + f->write(FSPI_INTR_IPCMDDONE_MASK, f->iobase + FSPI_INTR); > + > + if (reg & FSPI_INTR_IPCMDDONE_MASK) > + complete(&f->c); > + > + return IRQ_HANDLED; > +} [...] > +/* > + * If the slave device content being changed by Write/Erase, need to > + * invalidate the AHB buffer. This can be achieved by doing the reset > + * of controller after setting MCR0[SWRESET] bit. > + */ > +static inline void nxp_fspi_invalid(struct nxp_fspi *f) > +{ > + u32 reg; > + > + reg = f->read(f->iobase + FSPI_MCR0); > + f->write(reg | FSPI_MCR0_SWRST_MASK, f->iobase + FSPI_MCR0); > + > + while (f->read(f->iobase + FSPI_MCR0) & FSPI_MCR0_SWRST_MASK) > + ; Did you consider using readl_poll_timeout[_atomic]()? if (f->big_endian) mask = (u32)cpu_to_be32(FSPI_MCR0_SWRST_MASK); else mask = (u32)cpu_to_be32(FSPI_MCR0_SWRST_MASK); ret = readl_poll_timeout(f->iobase + FSPI_MCR0, reg, reg & mask, 0, FSPI_SWRST_TIMEOUT); WARN_ON(ret); > +} [...] > +static void nxp_fspi_read_ahb(struct nxp_fspi *f, const struct spi_mem_op *op) > +{ > + u32 len = op->data.nbytes; > + > + /* Read out the data directly from the AHB buffer. */ > + memcpy_fromio(op->data.buf.in, (f->ahb_addr + op->addr.val), len); Don't know if it's supported, but if it is, I recommend using DMA to do this copy, because otherwise you might stall the CPU for quite a long time if the flash is operating in a low-speed mode, and RT maintainers will complain about that at some point ;-). > +} > + > +static void nxp_fspi_fill_txfifo(struct nxp_fspi *f, > + const struct spi_mem_op *op) > +{ > + void __iomem *base = f->iobase; > + int i, j; > + int size, tmp_size, wm_size; > + u32 data = 0; > + u32 *txbuf = (u32 *) op->data.buf.out; > + > + /* clear the TX FIFO. */ > + f->write(FSPI_IPTXFCR_CLR_MASK, base + FSPI_IPTXFCR); > + > + /* Default value of water mark level is 8 bytes. */ > + wm_size = 8; > + size = op->data.nbytes / wm_size; > + for (i = 0; i < size; i++) { > + /* Wait for TXFIFO empty */ > + while (!(f->read(base + FSPI_INTR) & FSPI_INTR_IPTXWE_MASK)) > + ; Use readl_poll_timeout(), or even better, provide an helper (fspi_readl_poll_timeout()?) that hides the BE/LE stuff, so that you can reuse it when this pattern occurs. [...] > +static int nxp_fspi_exec_op(struct spi_mem *mem, const struct spi_mem_op *op) > +{ > + struct nxp_fspi *f = spi_controller_get_devdata(mem->spi->master); > + void __iomem *base = f->iobase; > + int err = 0; > + unsigned int timeout = 1000; > + > + mutex_lock(&f->lock); > + > + /* wait for the controller being ready */ > + do { > + u32 status; > + > + status = f->read(base + FSPI_STS0); > + if ((status & FSPI_STS0_ARB_IDLE_MASK) && > + (status & FSPI_STS0_SEQ_IDLE_MASK)) > + break; > + udelay(1); > + dev_dbg(f->dev, "The controller is busy, 0x%x\n", status); Same here. Note that I didn't spend time looking at how the IP works, which explains why I focus on tiny details here. Unfortunately, I won't have time to review the driver in more details, so I'll leave that to someone else, or let Mark decides if he's happy enough with the current version. Regards, Boris