]> git.proxmox.com Git - mirror_ubuntu-bionic-kernel.git/blame - drivers/mfd/sprd-sc27xx-spi.c
Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/jikos/hid
[mirror_ubuntu-bionic-kernel.git] / drivers / mfd / sprd-sc27xx-spi.c
CommitLineData
25ca4ae4
BW
1/*
2 * Copyright (C) 2017 Spreadtrum Communications Inc.
3 *
4 * This software is licensed under the terms of the GNU General Public
5 * License version 2, as published by the Free Software Foundation, and
6 * may be copied, distributed, and modified under those terms.
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 */
13
14#include <linux/interrupt.h>
15#include <linux/kernel.h>
16#include <linux/module.h>
17#include <linux/mfd/core.h>
18#include <linux/of_device.h>
19#include <linux/regmap.h>
20#include <linux/spi/spi.h>
21
22#define SPRD_PMIC_INT_MASK_STATUS 0x0
23#define SPRD_PMIC_INT_RAW_STATUS 0x4
24#define SPRD_PMIC_INT_EN 0x8
25
26#define SPRD_SC2731_IRQ_BASE 0x140
27#define SPRD_SC2731_IRQ_NUMS 16
28
29struct sprd_pmic {
30 struct regmap *regmap;
31 struct device *dev;
32 struct regmap_irq *irqs;
33 struct regmap_irq_chip irq_chip;
34 struct regmap_irq_chip_data *irq_data;
35 int irq;
36};
37
38struct sprd_pmic_data {
39 u32 irq_base;
40 u32 num_irqs;
41};
42
43/*
44 * Since different PMICs of SC27xx series can have different interrupt
45 * base address and irq number, we should save irq number and irq base
46 * in the device data structure.
47 */
48static const struct sprd_pmic_data sc2731_data = {
49 .irq_base = SPRD_SC2731_IRQ_BASE,
50 .num_irqs = SPRD_SC2731_IRQ_NUMS,
51};
52
53static const struct mfd_cell sprd_pmic_devs[] = {
54 {
55 .name = "sc27xx-wdt",
56 .of_compatible = "sprd,sc27xx-wdt",
57 }, {
58 .name = "sc27xx-rtc",
59 .of_compatible = "sprd,sc27xx-rtc",
60 }, {
61 .name = "sc27xx-charger",
62 .of_compatible = "sprd,sc27xx-charger",
63 }, {
64 .name = "sc27xx-chg-timer",
65 .of_compatible = "sprd,sc27xx-chg-timer",
66 }, {
67 .name = "sc27xx-fast-chg",
68 .of_compatible = "sprd,sc27xx-fast-chg",
69 }, {
70 .name = "sc27xx-chg-wdt",
71 .of_compatible = "sprd,sc27xx-chg-wdt",
72 }, {
73 .name = "sc27xx-typec",
74 .of_compatible = "sprd,sc27xx-typec",
75 }, {
76 .name = "sc27xx-flash",
77 .of_compatible = "sprd,sc27xx-flash",
78 }, {
79 .name = "sc27xx-eic",
80 .of_compatible = "sprd,sc27xx-eic",
81 }, {
82 .name = "sc27xx-efuse",
83 .of_compatible = "sprd,sc27xx-efuse",
84 }, {
85 .name = "sc27xx-thermal",
86 .of_compatible = "sprd,sc27xx-thermal",
87 }, {
88 .name = "sc27xx-adc",
89 .of_compatible = "sprd,sc27xx-adc",
90 }, {
91 .name = "sc27xx-audio-codec",
92 .of_compatible = "sprd,sc27xx-audio-codec",
93 }, {
94 .name = "sc27xx-regulator",
95 .of_compatible = "sprd,sc27xx-regulator",
96 }, {
97 .name = "sc27xx-vibrator",
98 .of_compatible = "sprd,sc27xx-vibrator",
99 }, {
100 .name = "sc27xx-keypad-led",
101 .of_compatible = "sprd,sc27xx-keypad-led",
102 }, {
103 .name = "sc27xx-bltc",
104 .of_compatible = "sprd,sc27xx-bltc",
105 }, {
106 .name = "sc27xx-fgu",
107 .of_compatible = "sprd,sc27xx-fgu",
108 }, {
109 .name = "sc27xx-7sreset",
110 .of_compatible = "sprd,sc27xx-7sreset",
111 }, {
112 .name = "sc27xx-poweroff",
113 .of_compatible = "sprd,sc27xx-poweroff",
114 },
115};
116
117static int sprd_pmic_spi_write(void *context, const void *data, size_t count)
118{
119 struct device *dev = context;
120 struct spi_device *spi = to_spi_device(dev);
121
122 return spi_write(spi, data, count);
123}
124
125static int sprd_pmic_spi_read(void *context,
126 const void *reg, size_t reg_size,
127 void *val, size_t val_size)
128{
129 struct device *dev = context;
130 struct spi_device *spi = to_spi_device(dev);
131 u32 rx_buf[2] = { 0 };
132 int ret;
133
134 /* Now we only support one PMIC register to read every time. */
135 if (reg_size != sizeof(u32) || val_size != sizeof(u32))
136 return -EINVAL;
137
138 /* Copy address to read from into first element of SPI buffer. */
139 memcpy(rx_buf, reg, sizeof(u32));
140 ret = spi_read(spi, rx_buf, 1);
141 if (ret < 0)
142 return ret;
143
144 memcpy(val, rx_buf, val_size);
145 return 0;
146}
147
148static struct regmap_bus sprd_pmic_regmap = {
149 .write = sprd_pmic_spi_write,
150 .read = sprd_pmic_spi_read,
151 .reg_format_endian_default = REGMAP_ENDIAN_NATIVE,
152 .val_format_endian_default = REGMAP_ENDIAN_NATIVE,
153};
154
155static const struct regmap_config sprd_pmic_config = {
156 .reg_bits = 32,
157 .val_bits = 32,
158 .reg_stride = 4,
159 .max_register = 0xffff,
160};
161
162static int sprd_pmic_probe(struct spi_device *spi)
163{
164 struct sprd_pmic *ddata;
165 const struct sprd_pmic_data *pdata;
166 int ret, i;
167
168 pdata = of_device_get_match_data(&spi->dev);
169 if (!pdata) {
170 dev_err(&spi->dev, "No matching driver data found\n");
171 return -EINVAL;
172 }
173
174 ddata = devm_kzalloc(&spi->dev, sizeof(*ddata), GFP_KERNEL);
175 if (!ddata)
176 return -ENOMEM;
177
178 ddata->regmap = devm_regmap_init(&spi->dev, &sprd_pmic_regmap,
179 &spi->dev, &sprd_pmic_config);
180 if (IS_ERR(ddata->regmap)) {
181 ret = PTR_ERR(ddata->regmap);
182 dev_err(&spi->dev, "Failed to allocate register map %d\n", ret);
183 return ret;
184 }
185
186 spi_set_drvdata(spi, ddata);
187 ddata->dev = &spi->dev;
188 ddata->irq = spi->irq;
189
190 ddata->irq_chip.name = dev_name(&spi->dev);
191 ddata->irq_chip.status_base =
192 pdata->irq_base + SPRD_PMIC_INT_MASK_STATUS;
193 ddata->irq_chip.mask_base = pdata->irq_base + SPRD_PMIC_INT_EN;
194 ddata->irq_chip.ack_base = 0;
195 ddata->irq_chip.num_regs = 1;
196 ddata->irq_chip.num_irqs = pdata->num_irqs;
197 ddata->irq_chip.mask_invert = true;
198
199 ddata->irqs = devm_kzalloc(&spi->dev, sizeof(struct regmap_irq) *
200 pdata->num_irqs, GFP_KERNEL);
201 if (!ddata->irqs)
202 return -ENOMEM;
203
204 ddata->irq_chip.irqs = ddata->irqs;
205 for (i = 0; i < pdata->num_irqs; i++) {
206 ddata->irqs[i].reg_offset = i / pdata->num_irqs;
207 ddata->irqs[i].mask = BIT(i % pdata->num_irqs);
208 }
209
210 ret = devm_regmap_add_irq_chip(&spi->dev, ddata->regmap, ddata->irq,
211 IRQF_ONESHOT | IRQF_NO_SUSPEND, 0,
212 &ddata->irq_chip, &ddata->irq_data);
213 if (ret) {
214 dev_err(&spi->dev, "Failed to add PMIC irq chip %d\n", ret);
215 return ret;
216 }
217
218 ret = devm_mfd_add_devices(&spi->dev, PLATFORM_DEVID_AUTO,
219 sprd_pmic_devs, ARRAY_SIZE(sprd_pmic_devs),
220 NULL, 0,
221 regmap_irq_get_domain(ddata->irq_data));
222 if (ret) {
223 dev_err(&spi->dev, "Failed to register device %d\n", ret);
224 return ret;
225 }
226
227 return 0;
228}
229
230static const struct of_device_id sprd_pmic_match[] = {
231 { .compatible = "sprd,sc2731", .data = &sc2731_data },
232 {},
233};
234MODULE_DEVICE_TABLE(of, sprd_pmic_match);
235
236static struct spi_driver sprd_pmic_driver = {
237 .driver = {
238 .name = "sc27xx-pmic",
239 .bus = &spi_bus_type,
240 .of_match_table = sprd_pmic_match,
241 },
242 .probe = sprd_pmic_probe,
243};
244
245static int __init sprd_pmic_init(void)
246{
247 return spi_register_driver(&sprd_pmic_driver);
248}
249subsys_initcall(sprd_pmic_init);
250
251static void __exit sprd_pmic_exit(void)
252{
253 spi_unregister_driver(&sprd_pmic_driver);
254}
255module_exit(sprd_pmic_exit);
256
257MODULE_LICENSE("GPL v2");
258MODULE_DESCRIPTION("Spreadtrum SC27xx PMICs driver");
259MODULE_AUTHOR("Baolin Wang <baolin.wang@spreadtrum.com>");