]> git.kernelconcepts.de Git - karo-tx-linux.git/blob - drivers/mfd/atmel-hlcdc.c
Merge remote-tracking branch 'mfd/for-mfd-next'
[karo-tx-linux.git] / drivers / mfd / atmel-hlcdc.c
1 /*
2  * Copyright (C) 2014 Free Electrons
3  * Copyright (C) 2014 Atmel
4  *
5  * Author: Boris BREZILLON <boris.brezillon@free-electrons.com>
6  *
7  * This program is free software; you can redistribute it and/or modify it
8  * under the terms of the GNU General Public License version 2 as published by
9  * the Free Software Foundation.
10  *
11  * This program is distributed in the hope that it will be useful, but WITHOUT
12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
14  * more details.
15  *
16  * You should have received a copy of the GNU General Public License along with
17  * this program.  If not, see <http://www.gnu.org/licenses/>.
18  */
19
20 #include <linux/clk.h>
21 #include <linux/iopoll.h>
22 #include <linux/mfd/atmel-hlcdc.h>
23 #include <linux/mfd/core.h>
24 #include <linux/module.h>
25 #include <linux/platform_device.h>
26 #include <linux/regmap.h>
27
28 #define ATMEL_HLCDC_REG_MAX             (0x4000 - 0x4)
29
30 struct atmel_hlcdc_regmap {
31         void __iomem *regs;
32 };
33
34 static const struct mfd_cell atmel_hlcdc_cells[] = {
35         {
36                 .name = "atmel-hlcdc-pwm",
37                 .of_compatible = "atmel,hlcdc-pwm",
38         },
39         {
40                 .name = "atmel-hlcdc-dc",
41                 .of_compatible = "atmel,hlcdc-display-controller",
42         },
43 };
44
45 static int regmap_atmel_hlcdc_reg_write(void *context, unsigned int reg,
46                                         unsigned int val)
47 {
48         struct atmel_hlcdc_regmap *hregmap = context;
49
50         if (reg <= ATMEL_HLCDC_DIS) {
51                 u32 status;
52
53                 readl_poll_timeout(hregmap->regs + ATMEL_HLCDC_SR, status,
54                                    !(status & ATMEL_HLCDC_SIP), 1, 100);
55         }
56
57         writel(val, hregmap->regs + reg);
58
59         return 0;
60 }
61
62 static int regmap_atmel_hlcdc_reg_read(void *context, unsigned int reg,
63                                        unsigned int *val)
64 {
65         struct atmel_hlcdc_regmap *hregmap = context;
66
67         *val = readl(hregmap->regs + reg);
68
69         return 0;
70 }
71
72 static const struct regmap_config atmel_hlcdc_regmap_config = {
73         .reg_bits = 32,
74         .val_bits = 32,
75         .reg_stride = 4,
76         .max_register = ATMEL_HLCDC_REG_MAX,
77         .reg_write = regmap_atmel_hlcdc_reg_write,
78         .reg_read = regmap_atmel_hlcdc_reg_read,
79         .fast_io = true,
80 };
81
82 static int atmel_hlcdc_probe(struct platform_device *pdev)
83 {
84         struct atmel_hlcdc_regmap *hregmap;
85         struct device *dev = &pdev->dev;
86         struct atmel_hlcdc *hlcdc;
87         struct resource *res;
88
89         hregmap = devm_kzalloc(dev, sizeof(*hregmap), GFP_KERNEL);
90         if (!hregmap)
91                 return -ENOMEM;
92
93         hlcdc = devm_kzalloc(dev, sizeof(*hlcdc), GFP_KERNEL);
94         if (!hlcdc)
95                 return -ENOMEM;
96
97         res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
98         hregmap->regs = devm_ioremap_resource(dev, res);
99         if (IS_ERR(hregmap->regs))
100                 return PTR_ERR(hregmap->regs);
101
102         hlcdc->irq = platform_get_irq(pdev, 0);
103         if (hlcdc->irq < 0)
104                 return hlcdc->irq;
105
106         hlcdc->periph_clk = devm_clk_get(dev, "periph_clk");
107         if (IS_ERR(hlcdc->periph_clk)) {
108                 dev_err(dev, "failed to get peripheral clock\n");
109                 return PTR_ERR(hlcdc->periph_clk);
110         }
111
112         hlcdc->sys_clk = devm_clk_get(dev, "sys_clk");
113         if (IS_ERR(hlcdc->sys_clk)) {
114                 dev_err(dev, "failed to get system clock\n");
115                 return PTR_ERR(hlcdc->sys_clk);
116         }
117
118         hlcdc->slow_clk = devm_clk_get(dev, "slow_clk");
119         if (IS_ERR(hlcdc->slow_clk)) {
120                 dev_err(dev, "failed to get slow clock\n");
121                 return PTR_ERR(hlcdc->slow_clk);
122         }
123
124         hlcdc->regmap = devm_regmap_init(dev, NULL, hregmap,
125                                          &atmel_hlcdc_regmap_config);
126         if (IS_ERR(hlcdc->regmap))
127                 return PTR_ERR(hlcdc->regmap);
128
129         dev_set_drvdata(dev, hlcdc);
130
131         return mfd_add_devices(dev, -1, atmel_hlcdc_cells,
132                                ARRAY_SIZE(atmel_hlcdc_cells),
133                                NULL, 0, NULL);
134 }
135
136 static int atmel_hlcdc_remove(struct platform_device *pdev)
137 {
138         mfd_remove_devices(&pdev->dev);
139
140         return 0;
141 }
142
143 static const struct of_device_id atmel_hlcdc_match[] = {
144         { .compatible = "atmel,at91sam9n12-hlcdc" },
145         { .compatible = "atmel,at91sam9x5-hlcdc" },
146         { .compatible = "atmel,sama5d2-hlcdc" },
147         { .compatible = "atmel,sama5d3-hlcdc" },
148         { .compatible = "atmel,sama5d4-hlcdc" },
149         { /* sentinel */ },
150 };
151 MODULE_DEVICE_TABLE(of, atmel_hlcdc_match);
152
153 static struct platform_driver atmel_hlcdc_driver = {
154         .probe = atmel_hlcdc_probe,
155         .remove = atmel_hlcdc_remove,
156         .driver = {
157                 .name = "atmel-hlcdc",
158                 .of_match_table = atmel_hlcdc_match,
159         },
160 };
161 module_platform_driver(atmel_hlcdc_driver);
162
163 MODULE_ALIAS("platform:atmel-hlcdc");
164 MODULE_AUTHOR("Boris Brezillon <boris.brezillon@free-electrons.com>");
165 MODULE_DESCRIPTION("Atmel HLCDC driver");
166 MODULE_LICENSE("GPL v2");