2 * AS3711 PMIC MFC driver
4 * Copyright (C) 2012 Renesas Electronics Corporation
5 * Author: Guennadi Liakhovetski, <g.liakhovetski@gmx.de>
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the version 2 of the GNU General Public License as
9 * published by the Free Software Foundation
12 #include <linux/device.h>
13 #include <linux/err.h>
14 #include <linux/i2c.h>
15 #include <linux/init.h>
16 #include <linux/kernel.h>
17 #include <linux/mfd/as3711.h>
18 #include <linux/mfd/core.h>
19 #include <linux/module.h>
21 #include <linux/regmap.h>
22 #include <linux/slab.h>
30 * Ok to have it static: it is only used during probing and multiple I2C devices
31 * cannot be probed simultaneously. Just make sure to avoid stale data.
33 static struct mfd_cell as3711_subdevs
[] = {
34 [AS3711_REGULATOR
] = {.name
= "as3711-regulator",},
35 [AS3711_BACKLIGHT
] = {.name
= "as3711-backlight",},
38 static bool as3711_volatile_reg(struct device
*dev
, unsigned int reg
)
41 case AS3711_GPIO_SIGNAL_IN
:
42 case AS3711_INTERRUPT_STATUS_1
:
43 case AS3711_INTERRUPT_STATUS_2
:
44 case AS3711_INTERRUPT_STATUS_3
:
45 case AS3711_CHARGER_STATUS_1
:
46 case AS3711_CHARGER_STATUS_2
:
47 case AS3711_REG_STATUS
:
53 static bool as3711_precious_reg(struct device
*dev
, unsigned int reg
)
56 case AS3711_INTERRUPT_STATUS_1
:
57 case AS3711_INTERRUPT_STATUS_2
:
58 case AS3711_INTERRUPT_STATUS_3
:
64 static bool as3711_readable_reg(struct device
*dev
, unsigned int reg
)
67 case AS3711_SD_1_VOLTAGE
:
68 case AS3711_SD_2_VOLTAGE
:
69 case AS3711_SD_3_VOLTAGE
:
70 case AS3711_SD_4_VOLTAGE
:
71 case AS3711_LDO_1_VOLTAGE
:
72 case AS3711_LDO_2_VOLTAGE
:
73 case AS3711_LDO_3_VOLTAGE
:
74 case AS3711_LDO_4_VOLTAGE
:
75 case AS3711_LDO_5_VOLTAGE
:
76 case AS3711_LDO_6_VOLTAGE
:
77 case AS3711_LDO_7_VOLTAGE
:
78 case AS3711_LDO_8_VOLTAGE
:
79 case AS3711_SD_CONTROL
:
80 case AS3711_GPIO_SIGNAL_OUT
:
81 case AS3711_GPIO_SIGNAL_IN
:
82 case AS3711_SD_CONTROL_1
:
83 case AS3711_SD_CONTROL_2
:
84 case AS3711_CURR_CONTROL
:
85 case AS3711_CURR1_VALUE
:
86 case AS3711_CURR2_VALUE
:
87 case AS3711_CURR3_VALUE
:
88 case AS3711_STEPUP_CONTROL_1
:
89 case AS3711_STEPUP_CONTROL_2
:
90 case AS3711_STEPUP_CONTROL_4
:
91 case AS3711_STEPUP_CONTROL_5
:
92 case AS3711_REG_STATUS
:
93 case AS3711_INTERRUPT_STATUS_1
:
94 case AS3711_INTERRUPT_STATUS_2
:
95 case AS3711_INTERRUPT_STATUS_3
:
96 case AS3711_CHARGER_STATUS_1
:
97 case AS3711_CHARGER_STATUS_2
:
98 case AS3711_ASIC_ID_1
:
99 case AS3711_ASIC_ID_2
:
105 static const struct regmap_config as3711_regmap_config
= {
108 .volatile_reg
= as3711_volatile_reg
,
109 .readable_reg
= as3711_readable_reg
,
110 .precious_reg
= as3711_precious_reg
,
111 .max_register
= AS3711_MAX_REG
,
112 .num_reg_defaults_raw
= AS3711_NUM_REGS
,
113 .cache_type
= REGCACHE_RBTREE
,
117 static const struct of_device_id as3711_of_match
[] = {
118 {.compatible
= "ams,as3711",},
121 MODULE_DEVICE_TABLE(of
, as3711_of_match
);
124 static int as3711_i2c_probe(struct i2c_client
*client
,
125 const struct i2c_device_id
*id
)
127 struct as3711
*as3711
;
128 struct as3711_platform_data
*pdata
;
129 unsigned int id1
, id2
;
132 if (!client
->dev
.of_node
) {
133 pdata
= dev_get_platdata(&client
->dev
);
135 dev_dbg(&client
->dev
, "Platform data not found\n");
137 pdata
= devm_kzalloc(&client
->dev
,
138 sizeof(*pdata
), GFP_KERNEL
);
143 as3711
= devm_kzalloc(&client
->dev
, sizeof(struct as3711
), GFP_KERNEL
);
147 as3711
->dev
= &client
->dev
;
148 i2c_set_clientdata(client
, as3711
);
151 dev_notice(&client
->dev
, "IRQ not supported yet\n");
153 as3711
->regmap
= devm_regmap_init_i2c(client
, &as3711_regmap_config
);
154 if (IS_ERR(as3711
->regmap
)) {
155 ret
= PTR_ERR(as3711
->regmap
);
156 dev_err(&client
->dev
,
157 "regmap initialization failed: %d\n", ret
);
161 ret
= regmap_read(as3711
->regmap
, AS3711_ASIC_ID_1
, &id1
);
163 ret
= regmap_read(as3711
->regmap
, AS3711_ASIC_ID_2
, &id2
);
165 dev_err(&client
->dev
, "regmap_read() failed: %d\n", ret
);
170 dev_info(as3711
->dev
, "AS3711 detected: %x:%x\n", id1
, id2
);
173 * We can reuse as3711_subdevs[],
174 * it will be copied in mfd_add_devices()
177 as3711_subdevs
[AS3711_REGULATOR
].platform_data
=
179 as3711_subdevs
[AS3711_REGULATOR
].pdata_size
=
180 sizeof(pdata
->regulator
);
181 as3711_subdevs
[AS3711_BACKLIGHT
].platform_data
=
183 as3711_subdevs
[AS3711_BACKLIGHT
].pdata_size
=
184 sizeof(pdata
->backlight
);
186 as3711_subdevs
[AS3711_REGULATOR
].platform_data
= NULL
;
187 as3711_subdevs
[AS3711_REGULATOR
].pdata_size
= 0;
188 as3711_subdevs
[AS3711_BACKLIGHT
].platform_data
= NULL
;
189 as3711_subdevs
[AS3711_BACKLIGHT
].pdata_size
= 0;
192 ret
= mfd_add_devices(as3711
->dev
, -1, as3711_subdevs
,
193 ARRAY_SIZE(as3711_subdevs
), NULL
, 0, NULL
);
195 dev_err(&client
->dev
, "add mfd devices failed: %d\n", ret
);
200 static int as3711_i2c_remove(struct i2c_client
*client
)
202 struct as3711
*as3711
= i2c_get_clientdata(client
);
204 mfd_remove_devices(as3711
->dev
);
208 static const struct i2c_device_id as3711_i2c_id
[] = {
209 {.name
= "as3711", .driver_data
= 0},
213 MODULE_DEVICE_TABLE(i2c
, as3711_i2c_id
);
215 static struct i2c_driver as3711_i2c_driver
= {
218 .of_match_table
= of_match_ptr(as3711_of_match
),
220 .probe
= as3711_i2c_probe
,
221 .remove
= as3711_i2c_remove
,
222 .id_table
= as3711_i2c_id
,
225 static int __init
as3711_i2c_init(void)
227 return i2c_add_driver(&as3711_i2c_driver
);
229 /* Initialise early */
230 subsys_initcall(as3711_i2c_init
);
232 static void __exit
as3711_i2c_exit(void)
234 i2c_del_driver(&as3711_i2c_driver
);
236 module_exit(as3711_i2c_exit
);
238 MODULE_AUTHOR("Guennadi Liakhovetski <g.liakhovetski@gmx.de>");
239 MODULE_DESCRIPTION("AS3711 PMIC driver");
240 MODULE_LICENSE("GPL v2");