blob: a1b7fab91dd4d836a2c13037c249f0d8b7c160c1 [file] [log] [blame]
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +05301/*
2 * tps6507x-regulator.c
3 *
4 * Regulator driver for TPS65073 PMIC
5 *
6 * Copyright (C) 2009 Texas Instrument Incorporated - http://www.ti.com/
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License as
10 * published by the Free Software Foundation version 2.
11 *
12 * This program is distributed "as is" WITHOUT ANY WARRANTY of any kind,
13 * whether express or implied; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * General Public License for more details.
16 */
17
18#include <linux/kernel.h>
19#include <linux/module.h>
20#include <linux/init.h>
21#include <linux/err.h>
22#include <linux/platform_device.h>
23#include <linux/regulator/driver.h>
24#include <linux/regulator/machine.h>
Anuj Aggarwal7d148312010-07-12 17:54:06 +053025#include <linux/regulator/tps6507x.h>
Vishwanathrao Badarkhe, Manish6116ad92013-01-24 16:25:18 +053026#include <linux/of.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090027#include <linux/slab.h>
Todd Fischerd183fcc2010-04-05 20:23:56 -060028#include <linux/mfd/tps6507x.h>
Vishwanathrao Badarkhe, Manish6116ad92013-01-24 16:25:18 +053029#include <linux/regulator/of_regulator.h>
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +053030
31/* DCDC's */
32#define TPS6507X_DCDC_1 0
33#define TPS6507X_DCDC_2 1
34#define TPS6507X_DCDC_3 2
35/* LDOs */
36#define TPS6507X_LDO_1 3
37#define TPS6507X_LDO_2 4
38
39#define TPS6507X_MAX_REG_ID TPS6507X_LDO_2
40
41/* Number of step-down converters available */
42#define TPS6507X_NUM_DCDC 3
43/* Number of LDO voltage regulators available */
44#define TPS6507X_NUM_LDO 2
45/* Number of total regulators available */
46#define TPS6507X_NUM_REGULATOR (TPS6507X_NUM_DCDC + TPS6507X_NUM_LDO)
47
Axel Lin055917a2012-05-20 10:39:12 +080048/* Supported voltage values for regulators (in microVolts) */
49static const unsigned int VDCDCx_VSEL_table[] = {
50 725000, 750000, 775000, 800000,
51 825000, 850000, 875000, 900000,
52 925000, 950000, 975000, 1000000,
53 1025000, 1050000, 1075000, 1100000,
54 1125000, 1150000, 1175000, 1200000,
55 1225000, 1250000, 1275000, 1300000,
56 1325000, 1350000, 1375000, 1400000,
57 1425000, 1450000, 1475000, 1500000,
58 1550000, 1600000, 1650000, 1700000,
59 1750000, 1800000, 1850000, 1900000,
60 1950000, 2000000, 2050000, 2100000,
61 2150000, 2200000, 2250000, 2300000,
62 2350000, 2400000, 2450000, 2500000,
63 2550000, 2600000, 2650000, 2700000,
64 2750000, 2800000, 2850000, 2900000,
65 3000000, 3100000, 3200000, 3300000,
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +053066};
67
Axel Lin055917a2012-05-20 10:39:12 +080068static const unsigned int LDO1_VSEL_table[] = {
69 1000000, 1100000, 1200000, 1250000,
70 1300000, 1350000, 1400000, 1500000,
71 1600000, 1800000, 2500000, 2750000,
72 2800000, 3000000, 3100000, 3300000,
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +053073};
74
Axel Lin93b07e72012-06-05 08:58:13 +080075/* The voltage mapping table for LDO2 is the same as VDCDCx */
76#define LDO2_VSEL_table VDCDCx_VSEL_table
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +053077
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +053078struct tps_info {
79 const char *name;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +053080 u8 table_len;
Axel Lin055917a2012-05-20 10:39:12 +080081 const unsigned int *table;
Anuj Aggarwal7d148312010-07-12 17:54:06 +053082
83 /* Does DCDC high or the low register defines output voltage? */
84 bool defdcdc_default;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +053085};
86
Anuj Aggarwal7d148312010-07-12 17:54:06 +053087static struct tps_info tps6507x_pmic_regs[] = {
Todd Fischer31dd6a22010-04-08 09:04:55 +020088 {
89 .name = "VDCDC1",
Todd Fischer31dd6a22010-04-08 09:04:55 +020090 .table_len = ARRAY_SIZE(VDCDCx_VSEL_table),
91 .table = VDCDCx_VSEL_table,
92 },
93 {
94 .name = "VDCDC2",
Todd Fischer31dd6a22010-04-08 09:04:55 +020095 .table_len = ARRAY_SIZE(VDCDCx_VSEL_table),
96 .table = VDCDCx_VSEL_table,
97 },
98 {
99 .name = "VDCDC3",
Todd Fischer31dd6a22010-04-08 09:04:55 +0200100 .table_len = ARRAY_SIZE(VDCDCx_VSEL_table),
101 .table = VDCDCx_VSEL_table,
102 },
103 {
104 .name = "LDO1",
Todd Fischer31dd6a22010-04-08 09:04:55 +0200105 .table_len = ARRAY_SIZE(LDO1_VSEL_table),
106 .table = LDO1_VSEL_table,
107 },
108 {
109 .name = "LDO2",
Todd Fischer31dd6a22010-04-08 09:04:55 +0200110 .table_len = ARRAY_SIZE(LDO2_VSEL_table),
111 .table = LDO2_VSEL_table,
112 },
113};
114
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600115struct tps6507x_pmic {
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530116 struct regulator_desc desc[TPS6507X_NUM_REGULATOR];
Todd Fischer31dd6a22010-04-08 09:04:55 +0200117 struct tps6507x_dev *mfd;
Anuj Aggarwal7d148312010-07-12 17:54:06 +0530118 struct tps_info *info[TPS6507X_NUM_REGULATOR];
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530119 struct mutex io_lock;
120};
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600121static inline int tps6507x_pmic_read(struct tps6507x_pmic *tps, u8 reg)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530122{
Todd Fischer31dd6a22010-04-08 09:04:55 +0200123 u8 val;
124 int err;
125
126 err = tps->mfd->read_dev(tps->mfd, reg, 1, &val);
127
128 if (err)
129 return err;
130
131 return val;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530132}
133
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600134static inline int tps6507x_pmic_write(struct tps6507x_pmic *tps, u8 reg, u8 val)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530135{
Todd Fischer31dd6a22010-04-08 09:04:55 +0200136 return tps->mfd->write_dev(tps->mfd, reg, 1, &val);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530137}
138
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600139static int tps6507x_pmic_set_bits(struct tps6507x_pmic *tps, u8 reg, u8 mask)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530140{
141 int err, data;
142
143 mutex_lock(&tps->io_lock);
144
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600145 data = tps6507x_pmic_read(tps, reg);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530146 if (data < 0) {
Todd Fischer31dd6a22010-04-08 09:04:55 +0200147 dev_err(tps->mfd->dev, "Read from reg 0x%x failed\n", reg);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530148 err = data;
149 goto out;
150 }
151
152 data |= mask;
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600153 err = tps6507x_pmic_write(tps, reg, data);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530154 if (err)
Todd Fischer31dd6a22010-04-08 09:04:55 +0200155 dev_err(tps->mfd->dev, "Write for reg 0x%x failed\n", reg);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530156
157out:
158 mutex_unlock(&tps->io_lock);
159 return err;
160}
161
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600162static int tps6507x_pmic_clear_bits(struct tps6507x_pmic *tps, u8 reg, u8 mask)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530163{
164 int err, data;
165
166 mutex_lock(&tps->io_lock);
167
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600168 data = tps6507x_pmic_read(tps, reg);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530169 if (data < 0) {
Todd Fischer31dd6a22010-04-08 09:04:55 +0200170 dev_err(tps->mfd->dev, "Read from reg 0x%x failed\n", reg);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530171 err = data;
172 goto out;
173 }
174
175 data &= ~mask;
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600176 err = tps6507x_pmic_write(tps, reg, data);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530177 if (err)
Todd Fischer31dd6a22010-04-08 09:04:55 +0200178 dev_err(tps->mfd->dev, "Write for reg 0x%x failed\n", reg);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530179
180out:
181 mutex_unlock(&tps->io_lock);
182 return err;
183}
184
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600185static int tps6507x_pmic_reg_read(struct tps6507x_pmic *tps, u8 reg)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530186{
187 int data;
188
189 mutex_lock(&tps->io_lock);
190
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600191 data = tps6507x_pmic_read(tps, reg);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530192 if (data < 0)
Todd Fischer31dd6a22010-04-08 09:04:55 +0200193 dev_err(tps->mfd->dev, "Read from reg 0x%x failed\n", reg);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530194
195 mutex_unlock(&tps->io_lock);
196 return data;
197}
198
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600199static int tps6507x_pmic_reg_write(struct tps6507x_pmic *tps, u8 reg, u8 val)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530200{
201 int err;
202
203 mutex_lock(&tps->io_lock);
204
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600205 err = tps6507x_pmic_write(tps, reg, val);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530206 if (err < 0)
Todd Fischer31dd6a22010-04-08 09:04:55 +0200207 dev_err(tps->mfd->dev, "Write for reg 0x%x failed\n", reg);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530208
209 mutex_unlock(&tps->io_lock);
210 return err;
211}
212
Axel Linf2933d32012-03-12 15:57:50 +0800213static int tps6507x_pmic_is_enabled(struct regulator_dev *dev)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530214{
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600215 struct tps6507x_pmic *tps = rdev_get_drvdata(dev);
Axel Linf2933d32012-03-12 15:57:50 +0800216 int data, rid = rdev_get_id(dev);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530217 u8 shift;
218
Axel Linf2933d32012-03-12 15:57:50 +0800219 if (rid < TPS6507X_DCDC_1 || rid > TPS6507X_LDO_2)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530220 return -EINVAL;
221
Axel Linf2933d32012-03-12 15:57:50 +0800222 shift = TPS6507X_MAX_REG_ID - rid;
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600223 data = tps6507x_pmic_reg_read(tps, TPS6507X_REG_CON_CTRL1);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530224
225 if (data < 0)
226 return data;
227 else
228 return (data & 1<<shift) ? 1 : 0;
229}
230
Axel Linf2933d32012-03-12 15:57:50 +0800231static int tps6507x_pmic_enable(struct regulator_dev *dev)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530232{
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600233 struct tps6507x_pmic *tps = rdev_get_drvdata(dev);
Axel Linf2933d32012-03-12 15:57:50 +0800234 int rid = rdev_get_id(dev);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530235 u8 shift;
236
Axel Linf2933d32012-03-12 15:57:50 +0800237 if (rid < TPS6507X_DCDC_1 || rid > TPS6507X_LDO_2)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530238 return -EINVAL;
239
Axel Linf2933d32012-03-12 15:57:50 +0800240 shift = TPS6507X_MAX_REG_ID - rid;
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600241 return tps6507x_pmic_set_bits(tps, TPS6507X_REG_CON_CTRL1, 1 << shift);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530242}
243
Axel Linf2933d32012-03-12 15:57:50 +0800244static int tps6507x_pmic_disable(struct regulator_dev *dev)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530245{
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600246 struct tps6507x_pmic *tps = rdev_get_drvdata(dev);
Axel Linf2933d32012-03-12 15:57:50 +0800247 int rid = rdev_get_id(dev);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530248 u8 shift;
249
Axel Linf2933d32012-03-12 15:57:50 +0800250 if (rid < TPS6507X_DCDC_1 || rid > TPS6507X_LDO_2)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530251 return -EINVAL;
252
Axel Linf2933d32012-03-12 15:57:50 +0800253 shift = TPS6507X_MAX_REG_ID - rid;
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600254 return tps6507x_pmic_clear_bits(tps, TPS6507X_REG_CON_CTRL1,
255 1 << shift);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530256}
257
Axel Lin7c842a12012-04-20 18:33:52 +0800258static int tps6507x_pmic_get_voltage_sel(struct regulator_dev *dev)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530259{
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600260 struct tps6507x_pmic *tps = rdev_get_drvdata(dev);
Axel Linf2933d32012-03-12 15:57:50 +0800261 int data, rid = rdev_get_id(dev);
262 u8 reg, mask;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530263
Axel Linf2933d32012-03-12 15:57:50 +0800264 switch (rid) {
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530265 case TPS6507X_DCDC_1:
266 reg = TPS6507X_REG_DEFDCDC1;
Axel Linf2933d32012-03-12 15:57:50 +0800267 mask = TPS6507X_DEFDCDCX_DCDC_MASK;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530268 break;
269 case TPS6507X_DCDC_2:
Axel Linf2933d32012-03-12 15:57:50 +0800270 if (tps->info[rid]->defdcdc_default)
Anuj Aggarwal7d148312010-07-12 17:54:06 +0530271 reg = TPS6507X_REG_DEFDCDC2_HIGH;
272 else
273 reg = TPS6507X_REG_DEFDCDC2_LOW;
Axel Linf2933d32012-03-12 15:57:50 +0800274 mask = TPS6507X_DEFDCDCX_DCDC_MASK;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530275 break;
276 case TPS6507X_DCDC_3:
Axel Linf2933d32012-03-12 15:57:50 +0800277 if (tps->info[rid]->defdcdc_default)
Anuj Aggarwal7d148312010-07-12 17:54:06 +0530278 reg = TPS6507X_REG_DEFDCDC3_HIGH;
279 else
280 reg = TPS6507X_REG_DEFDCDC3_LOW;
Axel Linf2933d32012-03-12 15:57:50 +0800281 mask = TPS6507X_DEFDCDCX_DCDC_MASK;
282 break;
283 case TPS6507X_LDO_1:
284 reg = TPS6507X_REG_LDO_CTRL1;
285 mask = TPS6507X_REG_LDO_CTRL1_LDO1_MASK;
286 break;
287 case TPS6507X_LDO_2:
288 reg = TPS6507X_REG_DEFLDO2;
289 mask = TPS6507X_REG_DEFLDO2_LDO2_MASK;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530290 break;
291 default:
292 return -EINVAL;
293 }
294
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600295 data = tps6507x_pmic_reg_read(tps, reg);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530296 if (data < 0)
297 return data;
298
Axel Linf2933d32012-03-12 15:57:50 +0800299 data &= mask;
Axel Lin7c842a12012-04-20 18:33:52 +0800300 return data;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530301}
302
Axel Linca61a7b2012-03-12 15:59:54 +0800303static int tps6507x_pmic_set_voltage_sel(struct regulator_dev *dev,
304 unsigned selector)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530305{
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600306 struct tps6507x_pmic *tps = rdev_get_drvdata(dev);
Axel Linca61a7b2012-03-12 15:59:54 +0800307 int data, rid = rdev_get_id(dev);
Axel Linf2933d32012-03-12 15:57:50 +0800308 u8 reg, mask;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530309
Axel Linf2933d32012-03-12 15:57:50 +0800310 switch (rid) {
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530311 case TPS6507X_DCDC_1:
312 reg = TPS6507X_REG_DEFDCDC1;
Axel Linf2933d32012-03-12 15:57:50 +0800313 mask = TPS6507X_DEFDCDCX_DCDC_MASK;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530314 break;
315 case TPS6507X_DCDC_2:
Axel Linf2933d32012-03-12 15:57:50 +0800316 if (tps->info[rid]->defdcdc_default)
Anuj Aggarwal7d148312010-07-12 17:54:06 +0530317 reg = TPS6507X_REG_DEFDCDC2_HIGH;
318 else
319 reg = TPS6507X_REG_DEFDCDC2_LOW;
Axel Linf2933d32012-03-12 15:57:50 +0800320 mask = TPS6507X_DEFDCDCX_DCDC_MASK;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530321 break;
322 case TPS6507X_DCDC_3:
Axel Linf2933d32012-03-12 15:57:50 +0800323 if (tps->info[rid]->defdcdc_default)
Anuj Aggarwal7d148312010-07-12 17:54:06 +0530324 reg = TPS6507X_REG_DEFDCDC3_HIGH;
325 else
326 reg = TPS6507X_REG_DEFDCDC3_LOW;
Axel Linf2933d32012-03-12 15:57:50 +0800327 mask = TPS6507X_DEFDCDCX_DCDC_MASK;
328 break;
329 case TPS6507X_LDO_1:
330 reg = TPS6507X_REG_LDO_CTRL1;
331 mask = TPS6507X_REG_LDO_CTRL1_LDO1_MASK;
332 break;
333 case TPS6507X_LDO_2:
334 reg = TPS6507X_REG_DEFLDO2;
335 mask = TPS6507X_REG_DEFLDO2_LDO2_MASK;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530336 break;
337 default:
338 return -EINVAL;
339 }
340
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600341 data = tps6507x_pmic_reg_read(tps, reg);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530342 if (data < 0)
343 return data;
344
345 data &= ~mask;
Axel Linca61a7b2012-03-12 15:59:54 +0800346 data |= selector;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530347
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600348 return tps6507x_pmic_reg_write(tps, reg, data);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530349}
350
Axel Lin646e2682019-03-28 20:02:23 +0800351static const struct regulator_ops tps6507x_pmic_ops = {
Axel Linf2933d32012-03-12 15:57:50 +0800352 .is_enabled = tps6507x_pmic_is_enabled,
353 .enable = tps6507x_pmic_enable,
354 .disable = tps6507x_pmic_disable,
Axel Lin7c842a12012-04-20 18:33:52 +0800355 .get_voltage_sel = tps6507x_pmic_get_voltage_sel,
Axel Linca61a7b2012-03-12 15:59:54 +0800356 .set_voltage_sel = tps6507x_pmic_set_voltage_sel,
Axel Lin055917a2012-05-20 10:39:12 +0800357 .list_voltage = regulator_list_voltage_table,
Axel Lina1bb63a2013-04-25 11:33:42 +0800358 .map_voltage = regulator_map_voltage_ascend,
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530359};
360
Axel Linf979c082019-03-28 20:02:25 +0800361static int tps6507x_pmic_of_parse_cb(struct device_node *np,
362 const struct regulator_desc *desc,
363 struct regulator_config *config)
Vishwanathrao Badarkhe, Manish6116ad92013-01-24 16:25:18 +0530364{
Axel Linf979c082019-03-28 20:02:25 +0800365 struct tps6507x_pmic *tps = config->driver_data;
366 struct tps_info *info = tps->info[desc->id];
367 u32 prop;
368 int ret;
Vishwanathrao Badarkhe, Manish6116ad92013-01-24 16:25:18 +0530369
Axel Linf979c082019-03-28 20:02:25 +0800370 ret = of_property_read_u32(np, "ti,defdcdc_default", &prop);
371 if (!ret)
372 info->defdcdc_default = prop;
Vishwanathrao Badarkhe, Manish6116ad92013-01-24 16:25:18 +0530373
Axel Linf979c082019-03-28 20:02:25 +0800374 return 0;
Vishwanathrao Badarkhe, Manish6116ad92013-01-24 16:25:18 +0530375}
Manish Badarkhe4246e552014-02-06 00:06:00 +0530376
Bill Pembertona5023572012-11-19 13:22:22 -0500377static int tps6507x_pmic_probe(struct platform_device *pdev)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530378{
Todd Fischer31dd6a22010-04-08 09:04:55 +0200379 struct tps6507x_dev *tps6507x_dev = dev_get_drvdata(pdev->dev.parent);
Anuj Aggarwal7d148312010-07-12 17:54:06 +0530380 struct tps_info *info = &tps6507x_pmic_regs[0];
Mark Brownc172708d2012-04-04 00:50:22 +0100381 struct regulator_config config = { };
Axel Linf979c082019-03-28 20:02:25 +0800382 struct regulator_init_data *init_data = NULL;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530383 struct regulator_dev *rdev;
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600384 struct tps6507x_pmic *tps;
Todd Fischer0bc20bb2010-04-05 20:23:57 -0600385 struct tps6507x_board *tps_board;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530386 int i;
387
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530388 /**
Todd Fischer0bc20bb2010-04-05 20:23:57 -0600389 * tps_board points to pmic related constants
390 * coming from the board-evm file.
391 */
392
Todd Fischer31dd6a22010-04-08 09:04:55 +0200393 tps_board = dev_get_platdata(tps6507x_dev->dev);
Axel Linf979c082019-03-28 20:02:25 +0800394 if (tps_board)
395 init_data = tps_board->tps6507x_pmic_init_data;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530396
Axel Lin9eb0c422012-04-11 14:40:18 +0800397 tps = devm_kzalloc(&pdev->dev, sizeof(*tps), GFP_KERNEL);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530398 if (!tps)
399 return -ENOMEM;
400
401 mutex_init(&tps->io_lock);
402
403 /* common for all regulators */
Todd Fischer31dd6a22010-04-08 09:04:55 +0200404 tps->mfd = tps6507x_dev;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530405
406 for (i = 0; i < TPS6507X_NUM_REGULATOR; i++, info++, init_data++) {
407 /* Register the regulators */
408 tps->info[i] = info;
Axel Linf979c082019-03-28 20:02:25 +0800409 if (init_data && init_data->driver_data) {
Anuj Aggarwal7d148312010-07-12 17:54:06 +0530410 struct tps6507x_reg_platform_data *data =
Manish Badarkhe4246e552014-02-06 00:06:00 +0530411 init_data->driver_data;
Axel Linf979c082019-03-28 20:02:25 +0800412 info->defdcdc_default = data->defdcdc_default;
Anuj Aggarwal7d148312010-07-12 17:54:06 +0530413 }
414
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530415 tps->desc[i].name = info->name;
Axel Linf979c082019-03-28 20:02:25 +0800416 tps->desc[i].of_match = of_match_ptr(info->name);
417 tps->desc[i].regulators_node = of_match_ptr("regulators");
418 tps->desc[i].of_parse_cb = tps6507x_pmic_of_parse_cb;
Axel Lin77fa44d2011-05-12 13:47:50 +0800419 tps->desc[i].id = i;
Axel Lin0fcdb102011-08-02 15:34:12 +0800420 tps->desc[i].n_voltages = info->table_len;
Axel Lin055917a2012-05-20 10:39:12 +0800421 tps->desc[i].volt_table = info->table;
Axel Linf2933d32012-03-12 15:57:50 +0800422 tps->desc[i].ops = &tps6507x_pmic_ops;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530423 tps->desc[i].type = REGULATOR_VOLTAGE;
424 tps->desc[i].owner = THIS_MODULE;
425
Mark Brownc172708d2012-04-04 00:50:22 +0100426 config.dev = tps6507x_dev->dev;
427 config.init_data = init_data;
428 config.driver_data = tps;
429
Sachin Kamat71b710e2013-09-04 17:17:47 +0530430 rdev = devm_regulator_register(&pdev->dev, &tps->desc[i],
431 &config);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530432 if (IS_ERR(rdev)) {
Todd Fischer31dd6a22010-04-08 09:04:55 +0200433 dev_err(tps6507x_dev->dev,
434 "failed to register %s regulator\n",
435 pdev->name);
Sachin Kamat71b710e2013-09-04 17:17:47 +0530436 return PTR_ERR(rdev);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530437 }
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530438 }
439
Todd Fischer31dd6a22010-04-08 09:04:55 +0200440 tps6507x_dev->pmic = tps;
Axel Lind7399fa2010-08-09 15:51:23 +0800441 platform_set_drvdata(pdev, tps6507x_dev);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530442
443 return 0;
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530444}
445
Todd Fischer31dd6a22010-04-08 09:04:55 +0200446static struct platform_driver tps6507x_pmic_driver = {
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530447 .driver = {
Todd Fischer31dd6a22010-04-08 09:04:55 +0200448 .name = "tps6507x-pmic",
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530449 },
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600450 .probe = tps6507x_pmic_probe,
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530451};
452
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600453static int __init tps6507x_pmic_init(void)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530454{
Todd Fischer31dd6a22010-04-08 09:04:55 +0200455 return platform_driver_register(&tps6507x_pmic_driver);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530456}
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600457subsys_initcall(tps6507x_pmic_init);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530458
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600459static void __exit tps6507x_pmic_cleanup(void)
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530460{
Todd Fischer31dd6a22010-04-08 09:04:55 +0200461 platform_driver_unregister(&tps6507x_pmic_driver);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530462}
Todd Fischer4ce5ba52010-04-05 20:23:58 -0600463module_exit(tps6507x_pmic_cleanup);
Anuj Aggarwal3fa5b8e2009-08-21 00:39:39 +0530464
465MODULE_AUTHOR("Texas Instruments");
466MODULE_DESCRIPTION("TPS6507x voltage regulator driver");
Liam Girdwood9e108d32009-08-24 10:31:34 +0100467MODULE_LICENSE("GPL v2");
Todd Fischer31dd6a22010-04-08 09:04:55 +0200468MODULE_ALIAS("platform:tps6507x-pmic");