On 16/11/15 12:01, Markus Pargmann wrote:
> This is the core driver for imx25 touchscreen/adc driver. The module
> has one shared ADC and two different conversion queues which use the
> ADC. The two queues are identical. Both can be used for general purpose
> ADC but one is meant to be used for touchscreens.
> 
> This driver is the core which manages the central components and
> registers of the TSC/ADC unit. It manages the IRQs and forwards them to
> the correct components.
> 
> Signed-off-by: Markus Pargmann <m...@pengutronix.de>
> Signed-off-by: Denis Carikli <de...@eukrea.com>
> 
> [ensure correct ADC clock depending on the IPG clock]
> Signed-off-by: Juergen Borleis <j...@pengutronix.de>
Looks good to me - one query on meaning of a comment inline.

Acked-by: Jonathan Cameron <ji...@kernel.org>

I'm taking the view this series wants to go through the MFD tree
but don't mind if it goes through IIO or input instead
if there is a good reason to do so.

Jonathan
> ---
> 
> Notes:
>     Changes in v7:
>      - Cleanup bit defines in header files to be more readable
>      - Fix irq check to return with an error for irq <= 0
>      - Add COMPILE_TEST in Kconfig file
>     
>     Changes in v5:
>      - Remove ifdef CONFIG_OF as this driver is only for DT usage
>      - Remove module owner
>      - Add Kconfig dependencies ARCH_MX25 and OF
>     
>     @Jonathan Cameron:
>     I left your acked-by on the patch as these were small changes. If it 
> should be
>     removed, please say so. Thanks
> 
>  drivers/mfd/Kconfig             |   9 ++
>  drivers/mfd/Makefile            |   2 +
>  drivers/mfd/fsl-imx25-tsadc.c   | 204 
> ++++++++++++++++++++++++++++++++++++++++
>  include/linux/mfd/imx25-tsadc.h | 140 +++++++++++++++++++++++++++
>  4 files changed, 355 insertions(+)
>  create mode 100644 drivers/mfd/fsl-imx25-tsadc.c
>  create mode 100644 include/linux/mfd/imx25-tsadc.h
> 
> diff --git a/drivers/mfd/Kconfig b/drivers/mfd/Kconfig
> index 4d92df6ef9fe..4222e202ad2b 100644
> --- a/drivers/mfd/Kconfig
> +++ b/drivers/mfd/Kconfig
> @@ -271,6 +271,15 @@ config MFD_MC13XXX_I2C
>       help
>         Select this if your MC13xxx is connected via an I2C bus.
>  
> +config MFD_MX25_TSADC
> +     tristate "Freescale i.MX25 integrated Touchscreen and ADC unit"
> +     select REGMAP_MMIO
> +     depends on (SOC_IMX25 && OF) || COMPILE_TEST
> +     help
> +       Enable support for the integrated Touchscreen and ADC unit of the
> +       i.MX25 processors. They consist of a conversion queue for general
> +       purpose ADC and a queue for Touchscreens.
> +
>  config MFD_HI6421_PMIC
>       tristate "HiSilicon Hi6421 PMU/Codec IC"
>       depends on OF
> diff --git a/drivers/mfd/Makefile b/drivers/mfd/Makefile
> index a8b76b81b467..5741be88c173 100644
> --- a/drivers/mfd/Makefile
> +++ b/drivers/mfd/Makefile
> @@ -81,6 +81,8 @@ obj-$(CONFIG_TWL4030_POWER)    += twl4030-power.o
>  obj-$(CONFIG_MFD_TWL4030_AUDIO)      += twl4030-audio.o
>  obj-$(CONFIG_TWL6040_CORE)   += twl6040.o
>  
> +obj-$(CONFIG_MFD_MX25_TSADC) += fsl-imx25-tsadc.o
> +
>  obj-$(CONFIG_MFD_MC13XXX)    += mc13xxx-core.o
>  obj-$(CONFIG_MFD_MC13XXX_SPI)        += mc13xxx-spi.o
>  obj-$(CONFIG_MFD_MC13XXX_I2C)        += mc13xxx-i2c.o
> diff --git a/drivers/mfd/fsl-imx25-tsadc.c b/drivers/mfd/fsl-imx25-tsadc.c
> new file mode 100644
> index 000000000000..e67d5ca81e10
> --- /dev/null
> +++ b/drivers/mfd/fsl-imx25-tsadc.c
> @@ -0,0 +1,204 @@
> +/*
> + * Copyright (C) 2014-2015 Pengutronix, Markus Pargmann <m...@pengutronix.de>
> + *
> + * This program is free software; you can redistribute it and/or modify it 
> under
> + * the terms of the GNU General Public License version 2 as published by the
> + * Free Software Foundation.
> + */
> +
> +#include <linux/clk.h>
> +#include <linux/interrupt.h>
> +#include <linux/irqchip/chained_irq.h>
> +#include <linux/irqdesc.h>
> +#include <linux/irqdomain.h>
> +#include <linux/irq.h>
> +#include <linux/mfd/imx25-tsadc.h>
> +#include <linux/module.h>
> +#include <linux/of.h>
> +#include <linux/of_platform.h>
> +#include <linux/platform_device.h>
> +#include <linux/regmap.h>
> +
> +static struct regmap_config mx25_tsadc_regmap_config = {
> +     .fast_io = true,
> +     .max_register = 8,
> +     .reg_bits = 32,
> +     .val_bits = 32,
> +     .reg_stride = 4,
> +};
> +
> +static void mx25_tsadc_irq_handler(struct irq_desc *desc)
> +{
> +     struct mx25_tsadc *tsadc = irq_desc_get_handler_data(desc);
> +     struct irq_chip *chip = irq_desc_get_chip(desc);
> +     u32 status;
> +
> +     chained_irq_enter(chip, desc);
> +
> +     regmap_read(tsadc->regs, MX25_TSC_TGSR, &status);
> +
> +     if (status & MX25_TGSR_GCQ_INT)
> +             generic_handle_irq(irq_find_mapping(tsadc->domain, 1));
> +
> +     if (status & MX25_TGSR_TCQ_INT)
> +             generic_handle_irq(irq_find_mapping(tsadc->domain, 0));
> +
> +     chained_irq_exit(chip, desc);
> +}
> +
> +static int mx25_tsadc_domain_map(struct irq_domain *d, unsigned int irq,
> +                              irq_hw_number_t hwirq)
> +{
> +     struct mx25_tsadc *tsadc = d->host_data;
> +
> +     irq_set_chip_data(irq, tsadc);
> +     irq_set_chip_and_handler(irq, &dummy_irq_chip,
> +                              handle_level_irq);
> +     irq_modify_status(irq, IRQ_NOREQUEST, IRQ_NOPROBE);
> +
> +     return 0;
> +}
> +
> +static struct irq_domain_ops mx25_tsadc_domain_ops = {
> +     .map = mx25_tsadc_domain_map,
> +     .xlate = irq_domain_xlate_onecell,
> +};
> +
> +static int mx25_tsadc_setup_irq(struct platform_device *pdev,
> +                             struct mx25_tsadc *tsadc)
> +{
> +     struct device *dev = &pdev->dev;
> +     struct device_node *np = dev->of_node;
> +     int irq;
> +
> +     irq = platform_get_irq(pdev, 0);
> +     if (irq <= 0) {
> +             dev_err(dev, "Failed to get irq\n");
> +             return irq;
> +     }
> +
> +     tsadc->domain = irq_domain_add_simple(np, 2, 0, &mx25_tsadc_domain_ops,
> +                                           tsadc);
> +     if (!tsadc->domain) {
> +             dev_err(dev, "Failed to add irq domain\n");
> +             return -ENOMEM;
> +     }
> +
> +     irq_set_chained_handler(irq, mx25_tsadc_irq_handler);
> +     irq_set_handler_data(irq, tsadc);
> +
> +     return 0;
> +}
> +
> +static void mx25_tsadc_setup_clk(struct platform_device *pdev,
> +                              struct mx25_tsadc *tsadc)
> +{
> +     unsigned clk_div;
> +
> +     /*
> +      * According to the datasheet the ADC clock should never
> +      * exceed 1,75 MHz. Base clock is the IPG and the ADC unit uses
> +      * a funny clock divider. To keep the time constant the ADC needs to do
> +      * one conversion, adapt the ADC internal clock divider to
> +      * the available IPG
I'm not entirely following this explanation...
maybe

...one conversion, then adapt the ADC...

> +      */
> +
> +     dev_dbg(&pdev->dev, "Found master clock at %lu Hz\n",
> +             clk_get_rate(tsadc->clk));
> +
> +     clk_div = DIV_ROUND_UP(clk_get_rate(tsadc->clk), 1750000);
> +     dev_dbg(&pdev->dev, "Setting up ADC clock divider to %u\n", clk_div);
> +
> +     /* adc clock = IPG clock / (2 * div + 2) */
> +     clk_div -= 2;
> +     clk_div /= 2;
> +
> +     /*
> +      * the ADC clock divider changes its behaviour when values below 4
> +      * are used: it is fixed to "/ 10" in this case
> +      */
> +     clk_div = max_t(unsigned, 4, clk_div);
> +
> +     dev_dbg(&pdev->dev, "Resulting ADC conversion clock at %lu Hz\n",
> +             clk_get_rate(tsadc->clk) / (2 * clk_div + 2));
> +
> +     regmap_update_bits(tsadc->regs, MX25_TSC_TGCR,
> +                        MX25_TGCR_ADCCLKCFG(0x1f),
> +                        MX25_TGCR_ADCCLKCFG(clk_div));
> +}
> +
> +static int mx25_tsadc_probe(struct platform_device *pdev)
> +{
> +     struct device *dev = &pdev->dev;
> +     struct device_node *np = dev->of_node;
> +     struct mx25_tsadc *tsadc;
> +     struct resource *res;
> +     int ret;
> +     void __iomem *iomem;
> +
> +     tsadc = devm_kzalloc(dev, sizeof(*tsadc), GFP_KERNEL);
> +     if (!tsadc)
> +             return -ENOMEM;
> +
> +     res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
> +     iomem = devm_ioremap_resource(dev, res);
> +     if (IS_ERR(iomem))
> +             return PTR_ERR(iomem);
> +
> +     tsadc->regs = devm_regmap_init_mmio(dev, iomem,
> +                                         &mx25_tsadc_regmap_config);
> +     if (IS_ERR(tsadc->regs)) {
> +             dev_err(dev, "Failed to initialize regmap\n");
> +             return PTR_ERR(tsadc->regs);
> +     }
> +
> +     tsadc->clk = devm_clk_get(dev, "ipg");
> +     if (IS_ERR(tsadc->clk)) {
> +             dev_err(dev, "Failed to get ipg clock\n");
> +             return PTR_ERR(tsadc->clk);
> +     }
> +
> +     /* setup clock according to the datasheet */
> +     mx25_tsadc_setup_clk(pdev, tsadc);
> +
> +     /* Enable clock and reset the component */
> +     regmap_update_bits(tsadc->regs, MX25_TSC_TGCR, MX25_TGCR_CLK_EN,
> +                        MX25_TGCR_CLK_EN);
> +     regmap_update_bits(tsadc->regs, MX25_TSC_TGCR, MX25_TGCR_TSC_RST,
> +                        MX25_TGCR_TSC_RST);
> +
> +     /* Setup powersaving mode, but enable internal reference voltage */
> +     regmap_update_bits(tsadc->regs, MX25_TSC_TGCR, MX25_TGCR_POWERMODE_MASK,
> +                        MX25_TGCR_POWERMODE_SAVE);
> +     regmap_update_bits(tsadc->regs, MX25_TSC_TGCR, MX25_TGCR_INTREFEN,
> +                        MX25_TGCR_INTREFEN);
> +
> +     ret = mx25_tsadc_setup_irq(pdev, tsadc);
> +     if (ret)
> +             return ret;
> +
> +     platform_set_drvdata(pdev, tsadc);
> +
> +     of_platform_populate(np, NULL, NULL, dev);
> +
> +     return 0;
> +}
> +
> +static const struct of_device_id mx25_tsadc_ids[] = {
> +     { .compatible = "fsl,imx25-tsadc" },
> +     { /* Sentinel */ }
> +};
> +
> +static struct platform_driver mx25_tsadc_driver = {
> +     .driver = {
> +             .name = "mx25-tsadc",
> +             .of_match_table = of_match_ptr(mx25_tsadc_ids),
> +     },
> +     .probe = mx25_tsadc_probe,
> +};
> +module_platform_driver(mx25_tsadc_driver);
> +
> +MODULE_DESCRIPTION("MFD for ADC/TSC for Freescale mx25");
> +MODULE_AUTHOR("Markus Pargmann <m...@pengutronix.de>");
> +MODULE_LICENSE("GPL v2");
> +MODULE_ALIAS("platform:mx25-tsadc");
> diff --git a/include/linux/mfd/imx25-tsadc.h b/include/linux/mfd/imx25-tsadc.h
> new file mode 100644
> index 000000000000..7fe4b8c3baac
> --- /dev/null
> +++ b/include/linux/mfd/imx25-tsadc.h
> @@ -0,0 +1,140 @@
> +#ifndef _LINUX_INCLUDE_MFD_IMX25_TSADC_H_
> +#define _LINUX_INCLUDE_MFD_IMX25_TSADC_H_
> +
> +struct regmap;
> +struct clk;
> +
> +struct mx25_tsadc {
> +     struct regmap *regs;
> +     struct irq_domain *domain;
> +     struct clk *clk;
> +};
> +
> +#define MX25_TSC_TGCR                        0x00
> +#define MX25_TSC_TGSR                        0x04
> +#define MX25_TSC_TICR                        0x08
> +
> +/* The same register layout for TC and GC queue */
> +#define MX25_ADCQ_FIFO                       0x00
> +#define MX25_ADCQ_CR                 0x04
> +#define MX25_ADCQ_SR                 0x08
> +#define MX25_ADCQ_MR                 0x0c
> +#define MX25_ADCQ_ITEM_7_0           0x20
> +#define MX25_ADCQ_ITEM_15_8          0x24
> +#define MX25_ADCQ_CFG(n)             (0x40 + ((n) * 0x4))
> +
> +#define MX25_ADCQ_MR_MASK            0xffffffff
> +
> +/* TGCR */
> +#define MX25_TGCR_PDBTIME(x)         ((x) << 25)
> +#define MX25_TGCR_PDBTIME_MASK               GENMASK(31, 25)
> +#define MX25_TGCR_PDBEN                      BIT(24)
> +#define MX25_TGCR_PDEN                       BIT(23)
> +#define MX25_TGCR_ADCCLKCFG(x)               ((x) << 16)
> +#define MX25_TGCR_GET_ADCCLK(x)              (((x) >> 16) & 0x1f)
> +#define MX25_TGCR_INTREFEN           BIT(10)
> +#define MX25_TGCR_POWERMODE_MASK     GENMASK(9, 8)
> +#define MX25_TGCR_POWERMODE_SAVE     (1 << 8)
> +#define MX25_TGCR_POWERMODE_ON               (2 << 8)
> +#define MX25_TGCR_STLC                       BIT(5)
> +#define MX25_TGCR_SLPC                       BIT(4)
> +#define MX25_TGCR_FUNC_RST           BIT(2)
> +#define MX25_TGCR_TSC_RST            BIT(1)
> +#define MX25_TGCR_CLK_EN             BIT(0)
> +
> +/* TGSR */
> +#define MX25_TGSR_SLP_INT            BIT(2)
> +#define MX25_TGSR_GCQ_INT            BIT(1)
> +#define MX25_TGSR_TCQ_INT            BIT(0)
> +
> +/* ADCQ_ITEM_* */
> +#define _MX25_ADCQ_ITEM(item, x)     ((x) << ((item) * 4))
> +#define MX25_ADCQ_ITEM(item, x)              ((item) >= 8 ? \
> +             _MX25_ADCQ_ITEM((item) - 8, (x)) : _MX25_ADCQ_ITEM((item), (x)))
> +
> +/* ADCQ_FIFO (TCQFIFO and GCQFIFO) */
> +#define MX25_ADCQ_FIFO_DATA(x)               (((x) >> 4) & 0xfff)
> +#define MX25_ADCQ_FIFO_ID(x)         ((x) & 0xf)
> +
> +/* ADCQ_CR (TCQR and GCQR) */
> +#define MX25_ADCQ_CR_PDCFG_LEVEL     BIT(19)
> +#define MX25_ADCQ_CR_PDMSK           BIT(18)
> +#define MX25_ADCQ_CR_FRST            BIT(17)
> +#define MX25_ADCQ_CR_QRST            BIT(16)
> +#define MX25_ADCQ_CR_RWAIT_MASK              GENMASK(15, 12)
> +#define MX25_ADCQ_CR_RWAIT(x)                ((x) << 12)
> +#define MX25_ADCQ_CR_WMRK_MASK               GENMASK(11, 8)
> +#define MX25_ADCQ_CR_WMRK(x)         ((x) << 8)
> +#define MX25_ADCQ_CR_LITEMID_MASK    (0xf << 4)
> +#define MX25_ADCQ_CR_LITEMID(x)              ((x) << 4)
> +#define MX25_ADCQ_CR_RPT             BIT(3)
> +#define MX25_ADCQ_CR_FQS             BIT(2)
> +#define MX25_ADCQ_CR_QSM_MASK                GENMASK(1, 0)
> +#define MX25_ADCQ_CR_QSM_PD          0x1
> +#define MX25_ADCQ_CR_QSM_FQS         0x2
> +#define MX25_ADCQ_CR_QSM_FQS_PD              0x3
> +
> +/* ADCQ_SR (TCQSR and GCQSR) */
> +#define MX25_ADCQ_SR_FDRY            BIT(15)
> +#define MX25_ADCQ_SR_FULL            BIT(14)
> +#define MX25_ADCQ_SR_EMPT            BIT(13)
> +#define MX25_ADCQ_SR_FDN(x)          (((x) >> 8) & 0x1f)
> +#define MX25_ADCQ_SR_FRR             BIT(6)
> +#define MX25_ADCQ_SR_FUR             BIT(5)
> +#define MX25_ADCQ_SR_FOR             BIT(4)
> +#define MX25_ADCQ_SR_EOQ             BIT(1)
> +#define MX25_ADCQ_SR_PD                      BIT(0)
> +
> +/* ADCQ_MR (TCQMR and GCQMR) */
> +#define MX25_ADCQ_MR_FDRY_DMA                BIT(31)
> +#define MX25_ADCQ_MR_FER_DMA         BIT(22)
> +#define MX25_ADCQ_MR_FUR_DMA         BIT(21)
> +#define MX25_ADCQ_MR_FOR_DMA         BIT(20)
> +#define MX25_ADCQ_MR_EOQ_DMA         BIT(17)
> +#define MX25_ADCQ_MR_PD_DMA          BIT(16)
> +#define MX25_ADCQ_MR_FDRY_IRQ                BIT(15)
> +#define MX25_ADCQ_MR_FER_IRQ         BIT(6)
> +#define MX25_ADCQ_MR_FUR_IRQ         BIT(5)
> +#define MX25_ADCQ_MR_FOR_IRQ         BIT(4)
> +#define MX25_ADCQ_MR_EOQ_IRQ         BIT(1)
> +#define MX25_ADCQ_MR_PD_IRQ          BIT(0)
> +
> +/* ADCQ_CFG (TICR, TCC0-7,GCC0-7) */
> +#define MX25_ADCQ_CFG_SETTLING_TIME(x)       ((x) << 24)
> +#define MX25_ADCQ_CFG_IGS            (1 << 20)
> +#define MX25_ADCQ_CFG_NOS_MASK               GENMASK(19, 16)
> +#define MX25_ADCQ_CFG_NOS(x)         (((x) - 1) << 16)
> +#define MX25_ADCQ_CFG_WIPER          (1 << 15)
> +#define MX25_ADCQ_CFG_YNLR           (1 << 14)
> +#define MX25_ADCQ_CFG_YPLL_HIGH              (0 << 12)
> +#define MX25_ADCQ_CFG_YPLL_OFF               (1 << 12)
> +#define MX25_ADCQ_CFG_YPLL_LOW               (3 << 12)
> +#define MX25_ADCQ_CFG_XNUR_HIGH              (0 << 10)
> +#define MX25_ADCQ_CFG_XNUR_OFF               (1 << 10)
> +#define MX25_ADCQ_CFG_XNUR_LOW               (3 << 10)
> +#define MX25_ADCQ_CFG_XPUL_HIGH              (0 << 9)
> +#define MX25_ADCQ_CFG_XPUL_OFF               (1 << 9)
> +#define MX25_ADCQ_CFG_REFP(sel)              ((sel) << 7)
> +#define MX25_ADCQ_CFG_REFP_YP                MX25_ADCQ_CFG_REFP(0)
> +#define MX25_ADCQ_CFG_REFP_XP                MX25_ADCQ_CFG_REFP(1)
> +#define MX25_ADCQ_CFG_REFP_EXT               MX25_ADCQ_CFG_REFP(2)
> +#define MX25_ADCQ_CFG_REFP_INT               MX25_ADCQ_CFG_REFP(3)
> +#define MX25_ADCQ_CFG_REFP_MASK              GENMASK(8, 7)
> +#define MX25_ADCQ_CFG_IN(sel)                ((sel) << 4)
> +#define MX25_ADCQ_CFG_IN_XP          MX25_ADCQ_CFG_IN(0)
> +#define MX25_ADCQ_CFG_IN_YP          MX25_ADCQ_CFG_IN(1)
> +#define MX25_ADCQ_CFG_IN_XN          MX25_ADCQ_CFG_IN(2)
> +#define MX25_ADCQ_CFG_IN_YN          MX25_ADCQ_CFG_IN(3)
> +#define MX25_ADCQ_CFG_IN_WIPER               MX25_ADCQ_CFG_IN(4)
> +#define MX25_ADCQ_CFG_IN_AUX0                MX25_ADCQ_CFG_IN(5)
> +#define MX25_ADCQ_CFG_IN_AUX1                MX25_ADCQ_CFG_IN(6)
> +#define MX25_ADCQ_CFG_IN_AUX2                MX25_ADCQ_CFG_IN(7)
> +#define MX25_ADCQ_CFG_REFN(sel)              ((sel) << 2)
> +#define MX25_ADCQ_CFG_REFN_XN                MX25_ADCQ_CFG_REFN(0)
> +#define MX25_ADCQ_CFG_REFN_YN                MX25_ADCQ_CFG_REFN(1)
> +#define MX25_ADCQ_CFG_REFN_NGND              MX25_ADCQ_CFG_REFN(2)
> +#define MX25_ADCQ_CFG_REFN_NGND2     MX25_ADCQ_CFG_REFN(3)
> +#define MX25_ADCQ_CFG_REFN_MASK              GENMASK(3, 2)
> +#define MX25_ADCQ_CFG_PENIACK                (1 << 1)
> +
> +#endif  /* _LINUX_INCLUDE_MFD_IMX25_TSADC_H_ */
> 

--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to