On Thu, 31 Jul 2014, Thor Thayer wrote: > On 07/31/2014 03:26 AM, Lee Jones wrote: > >On Wed, 30 Jul 2014, tthayer@xxxxxxxxxxxxxxxxxxxxx wrote: > > > >>From: Thor Thayer <tthayer@xxxxxxxxxxxxxxxxxxxxx> > >> > >>Add a simple MFD for the Altera SDRAM Controller. > >> > >>Signed-off-by: Alan Tull <atull@xxxxxxxxxxxxxxxxxxxxx> > >>Signed-off-by: Thor Thayer <tthayer@xxxxxxxxxxxxxxxxxxxxx> > >>--- > >>v1-8: The MFD implementation was not included in the original series. > >> > >>v9: New MFD implementation. > >>--- > >> MAINTAINERS | 5 ++ > >> drivers/mfd/Kconfig | 7 ++ > >> drivers/mfd/Makefile | 1 + > >> drivers/mfd/altera-sdr.c | 162 ++++++++++++++++++++++++++++++++++++++++ > >> include/linux/mfd/altera-sdr.h | 102 +++++++++++++++++++++++++ > >> 5 files changed, 277 insertions(+) > >> create mode 100644 drivers/mfd/altera-sdr.c > >> create mode 100644 include/linux/mfd/altera-sdr.h [...] > >>+++ b/drivers/mfd/altera-sdr.c > >>@@ -0,0 +1,162 @@ > >>+/* > >>+ * SDRAM Controller (SDR) MFD > >>+ * > >>+ * Copyright (C) 2014 Altera Corporation > >>+ * > >>+ * This program is free software; you can redistribute it and/or modify it > >>+ * under the terms and conditions of the GNU General Public License, > >>+ * version 2, as published by the Free Software Foundation. > >>+ * > >>+ * This program is distributed in the hope it will be useful, but WITHOUT > >>+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or > >>+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for > >>+ * more details. > >>+ * > >>+ * You should have received a copy of the GNU General Public License along with > >>+ * this program. If not, see <http://www.gnu.org/licenses/>. > >Can you use the shorter version of the licence? > Hi. This seems to be the shorter version of the license agreement > and is fairly common in the kernel, right? This is the long(ish) version. Many programs have the Copyright line and the "This program is free software;" line. It'll also be a good idea to keep the link to the full licence. [...] > >>+ { > >>+ .name = "altr_sdram_edac", > >>+ .of_compatible = "altr,sdram-edac", > >What other devices will there be? > > > There will be an FPGA bridge and a power control driver that will > need access to the SDR Controller registers. Okay. Do you know when they'll be upstreamed? > >>+u32 altera_sdr_readl(struct altera_sdr *sdr, u32 reg_offset) > >>+{ > >>+ return readl(sdr->reg_base + reg_offset); > >>+} > >>+EXPORT_SYMBOL_GPL(altera_sdr_readl); > >>+ > >>+void altera_sdr_writel(struct altera_sdr *sdr, u32 reg_offset, u32 value) > >>+{ > >>+ writel(value, sdr->reg_base + reg_offset); > >>+} > >>+EXPORT_SYMBOL_GPL(altera_sdr_writel); > >Why are you abstracting these? > > > >Might be better to use Regmap even. > regmap seems unnecessarily complex for what we're doing which is why > this method was chosen. > > Future drivers will access different sets of registers in the > device. These drivers won't share bitfields in the same register so > the MFD seemed like the best solution. Originally we implemented > this using syscon but that seems to be frowned upon so we changed to > using a MFD. Why was the use of syscon frowned upon? Can you link me to the thread? Writing directly to the registers sounds to me a lot worse than using infrastructure which was designed for these kinds of accesses. If you do choose to fiddle with the registers in this manner, is there any reason why you're calling back into here, rather than using readl() and writel() directly? > >>+u32 altera_sdr_mem_size(struct altera_sdr *sdr) > >>+{ > >>+ u32 size; > >>+ u32 read_reg, row, bank, col, cs, width; > >Weird that size is on its own. Either place on a single line or > >separate them all. > OK. I will change this. > >>+ read_reg = altera_sdr_readl(sdr, SDR_DRAMADDRW_OFST); > >>+ if (read_reg < 0) > >>+ return 0; > >>+ > >>+ width = altera_sdr_readl(sdr, SDR_DRAMIFWIDTH_OFST); > >>+ if (width < 0) > >>+ return 0; u32s cant be < 0. The 'u' means 'unsigned'. > >>+ col = (read_reg & SDR_DRAMADDRW_COLBITS_MASK) >> > >>+ SDR_DRAMADDRW_COLBITS_LSB; > >>+ row = (read_reg & SDR_DRAMADDRW_ROWBITS_MASK) >> > >>+ SDR_DRAMADDRW_ROWBITS_LSB; > >>+ bank = (read_reg & SDR_DRAMADDRW_BANKBITS_MASK) >> > >>+ SDR_DRAMADDRW_BANKBITS_LSB; > >>+ cs = (read_reg & SDR_DRAMADDRW_CSBITS_MASK) >> > >>+ SDR_DRAMADDRW_CSBITS_LSB; > >These would probably be better as macros. > > > OK. I will fix this. > >>+ /* Correct for ECC as its not addressible */ > >>+ if (width == SDR_DRAMIFWIDTH_32B_ECC) > >>+ width = 32; > >>+ if (width == SDR_DRAMIFWIDTH_16B_ECC) > >>+ width = 16; > >>+ > >>+ /* calculate the SDRAM size base on this info */ > >>+ size = 1 << (row + bank + col); > >>+ size = size * cs * (width / 8); > >>+ return size; > >>+} > >>+EXPORT_SYMBOL_GPL(altera_sdr_mem_size); > >Should this really be done in here? Isn't this an SDRAM function? > > > This register is part of the SDRAM controller and size information > may be required by the other drivers that share this memory > area/need SDRAM information. Then export a function from the SDRAM driver, not from here. [...] > >>+static const struct platform_device_id altera_sdr_ids[] = { > >>+ { "altera_sdr", }, > >>+ { } > >>+}; > >What's this for? > We don't strictly need it because we are driven by the device tree. > It can be removed it if is a problem but I'm not clear why it is a > problem. It's a problem because it's unused, superfluous bumph. :) [...] > >>+static int __init altera_sdr_init(void) > >>+{ > >>+ return platform_driver_register(&altera_sdr_driver); > >>+} > >>+postcore_initcall(altera_sdr_init); > >Why was this chosen? > We want this to happen pretty early. If you _need_ this is happen early, core_initcall() is more commonly used, but _why_ do you need it to happen this early? > >>+/* > >>+ * Copyright (C) 2014 Altera Corporation > >>+ * > >>+ * This program is free software; you can redistribute it and/or modify it > >>+ * under the terms and conditions of the GNU General Public License, > >>+ * version 2, as published by the Free Software Foundation. > >>+ * > >>+ * This program is distributed in the hope it will be useful, but WITHOUT > >>+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or > >>+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for > >>+ * more details. > >>+ * > >>+ * You should have received a copy of the GNU General Public License along with > >>+ * this program. If not, see <http://www.gnu.org/licenses/>. > >Use the short version. > Same as above. This is the shorter version that we've been using at Altera. If you read COPYING, you only require the Copyright line and a link to the full licence. In this case I'll be happy if you left in the "This program is free software;" line as well, but remove the paragraph below it. [...] > >>+/* Register access API */ > >>+u32 altera_sdr_readl(struct altera_sdr *sdr, u32 reg_offset); > >>+void altera_sdr_writel(struct altera_sdr *sdr, u32 reg_offset, u32 value); > >Regmap? > Same as above. This seems to be more complex than we need. I'm not sure I believe that. Regmap is as complex or as simple as you need. > Thanks for your comments and for reviewing! > > Thor You're welcome. OOI, do you own a hammer? ;) -- Lee Jones Linaro STMicroelectronics Landing Team Lead Linaro.org │ Open source software for ARM SoCs Follow Linaro: Facebook | Twitter | Blog -- To unsubscribe from this list: send the line "unsubscribe linux-doc" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html