Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 1 | /* |
| 2 | * TI LMU (Lighting Management Unit) Core Driver |
| 3 | * |
| 4 | * Copyright 2017 Texas Instruments |
| 5 | * |
| 6 | * Author: Milo Kim <milo.kim@ti.com> |
| 7 | * |
| 8 | * This program is free software; you can redistribute it and/or modify |
| 9 | * it under the terms of the GNU General Public License version 2 as |
| 10 | * published by the Free Software Foundation. |
| 11 | */ |
| 12 | |
| 13 | #include <linux/delay.h> |
| 14 | #include <linux/err.h> |
Pavel Machek | 7a6a395 | 2018-09-11 23:01:33 +0200 | [diff] [blame] | 15 | #include <linux/gpio/consumer.h> |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 16 | #include <linux/i2c.h> |
| 17 | #include <linux/kernel.h> |
| 18 | #include <linux/mfd/core.h> |
| 19 | #include <linux/mfd/ti-lmu.h> |
| 20 | #include <linux/mfd/ti-lmu-register.h> |
| 21 | #include <linux/module.h> |
| 22 | #include <linux/of.h> |
| 23 | #include <linux/of_device.h> |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 24 | #include <linux/slab.h> |
| 25 | |
| 26 | struct ti_lmu_data { |
Sebastian Reichel | 5b6850f | 2018-08-29 11:31:09 +0200 | [diff] [blame] | 27 | const struct mfd_cell *cells; |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 28 | int num_cells; |
| 29 | unsigned int max_register; |
| 30 | }; |
| 31 | |
| 32 | static int ti_lmu_enable_hw(struct ti_lmu *lmu, enum ti_lmu_id id) |
| 33 | { |
Pavel Machek | 7a6a395 | 2018-09-11 23:01:33 +0200 | [diff] [blame] | 34 | if (lmu->en_gpio) |
| 35 | gpiod_set_value(lmu->en_gpio, 1); |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 36 | |
| 37 | /* Delay about 1ms after HW enable pin control */ |
| 38 | usleep_range(1000, 1500); |
| 39 | |
| 40 | /* LM3631 has additional power up sequence - enable LCD_EN bit. */ |
| 41 | if (id == LM3631) { |
| 42 | return regmap_update_bits(lmu->regmap, LM3631_REG_DEVCTRL, |
| 43 | LM3631_LCD_EN_MASK, |
| 44 | LM3631_LCD_EN_MASK); |
| 45 | } |
| 46 | |
| 47 | return 0; |
| 48 | } |
| 49 | |
Pavel Machek | 7891d37 | 2018-09-11 23:03:44 +0200 | [diff] [blame^] | 50 | static void ti_lmu_disable_hw(void *data) |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 51 | { |
Pavel Machek | 7891d37 | 2018-09-11 23:03:44 +0200 | [diff] [blame^] | 52 | struct ti_lmu *lmu = data; |
Pavel Machek | 7a6a395 | 2018-09-11 23:01:33 +0200 | [diff] [blame] | 53 | if (lmu->en_gpio) |
| 54 | gpiod_set_value(lmu->en_gpio, 0); |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 55 | } |
| 56 | |
Sebastian Reichel | 5b6850f | 2018-08-29 11:31:09 +0200 | [diff] [blame] | 57 | static const struct mfd_cell lm3532_devices[] = { |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 58 | { |
| 59 | .name = "ti-lmu-backlight", |
| 60 | .id = LM3532, |
| 61 | .of_compatible = "ti,lm3532-backlight", |
| 62 | }, |
| 63 | }; |
| 64 | |
| 65 | #define LM363X_REGULATOR(_id) \ |
| 66 | { \ |
| 67 | .name = "lm363x-regulator", \ |
| 68 | .id = _id, \ |
| 69 | .of_compatible = "ti,lm363x-regulator", \ |
| 70 | } \ |
| 71 | |
Sebastian Reichel | 5b6850f | 2018-08-29 11:31:09 +0200 | [diff] [blame] | 72 | static const struct mfd_cell lm3631_devices[] = { |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 73 | LM363X_REGULATOR(LM3631_BOOST), |
| 74 | LM363X_REGULATOR(LM3631_LDO_CONT), |
| 75 | LM363X_REGULATOR(LM3631_LDO_OREF), |
| 76 | LM363X_REGULATOR(LM3631_LDO_POS), |
| 77 | LM363X_REGULATOR(LM3631_LDO_NEG), |
| 78 | { |
| 79 | .name = "ti-lmu-backlight", |
| 80 | .id = LM3631, |
| 81 | .of_compatible = "ti,lm3631-backlight", |
| 82 | }, |
| 83 | }; |
| 84 | |
Sebastian Reichel | 5b6850f | 2018-08-29 11:31:09 +0200 | [diff] [blame] | 85 | static const struct mfd_cell lm3632_devices[] = { |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 86 | LM363X_REGULATOR(LM3632_BOOST), |
| 87 | LM363X_REGULATOR(LM3632_LDO_POS), |
| 88 | LM363X_REGULATOR(LM3632_LDO_NEG), |
| 89 | { |
| 90 | .name = "ti-lmu-backlight", |
| 91 | .id = LM3632, |
| 92 | .of_compatible = "ti,lm3632-backlight", |
| 93 | }, |
| 94 | }; |
| 95 | |
Sebastian Reichel | 5b6850f | 2018-08-29 11:31:09 +0200 | [diff] [blame] | 96 | static const struct mfd_cell lm3633_devices[] = { |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 97 | { |
| 98 | .name = "ti-lmu-backlight", |
| 99 | .id = LM3633, |
| 100 | .of_compatible = "ti,lm3633-backlight", |
| 101 | }, |
| 102 | { |
| 103 | .name = "lm3633-leds", |
| 104 | .of_compatible = "ti,lm3633-leds", |
| 105 | }, |
| 106 | /* Monitoring driver for open/short circuit detection */ |
| 107 | { |
| 108 | .name = "ti-lmu-fault-monitor", |
| 109 | .id = LM3633, |
| 110 | .of_compatible = "ti,lm3633-fault-monitor", |
| 111 | }, |
| 112 | }; |
| 113 | |
Sebastian Reichel | 5b6850f | 2018-08-29 11:31:09 +0200 | [diff] [blame] | 114 | static const struct mfd_cell lm3695_devices[] = { |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 115 | { |
| 116 | .name = "ti-lmu-backlight", |
| 117 | .id = LM3695, |
| 118 | .of_compatible = "ti,lm3695-backlight", |
| 119 | }, |
| 120 | }; |
| 121 | |
Sebastian Reichel | 5b6850f | 2018-08-29 11:31:09 +0200 | [diff] [blame] | 122 | static const struct mfd_cell lm3697_devices[] = { |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 123 | { |
| 124 | .name = "ti-lmu-backlight", |
| 125 | .id = LM3697, |
| 126 | .of_compatible = "ti,lm3697-backlight", |
| 127 | }, |
| 128 | /* Monitoring driver for open/short circuit detection */ |
| 129 | { |
| 130 | .name = "ti-lmu-fault-monitor", |
| 131 | .id = LM3697, |
| 132 | .of_compatible = "ti,lm3697-fault-monitor", |
| 133 | }, |
| 134 | }; |
| 135 | |
| 136 | #define TI_LMU_DATA(chip, max_reg) \ |
| 137 | static const struct ti_lmu_data chip##_data = \ |
| 138 | { \ |
| 139 | .cells = chip##_devices, \ |
| 140 | .num_cells = ARRAY_SIZE(chip##_devices),\ |
| 141 | .max_register = max_reg, \ |
| 142 | } \ |
| 143 | |
| 144 | TI_LMU_DATA(lm3532, LM3532_MAX_REG); |
| 145 | TI_LMU_DATA(lm3631, LM3631_MAX_REG); |
| 146 | TI_LMU_DATA(lm3632, LM3632_MAX_REG); |
| 147 | TI_LMU_DATA(lm3633, LM3633_MAX_REG); |
| 148 | TI_LMU_DATA(lm3695, LM3695_MAX_REG); |
| 149 | TI_LMU_DATA(lm3697, LM3697_MAX_REG); |
| 150 | |
| 151 | static const struct of_device_id ti_lmu_of_match[] = { |
| 152 | { .compatible = "ti,lm3532", .data = &lm3532_data }, |
| 153 | { .compatible = "ti,lm3631", .data = &lm3631_data }, |
| 154 | { .compatible = "ti,lm3632", .data = &lm3632_data }, |
| 155 | { .compatible = "ti,lm3633", .data = &lm3633_data }, |
| 156 | { .compatible = "ti,lm3695", .data = &lm3695_data }, |
| 157 | { .compatible = "ti,lm3697", .data = &lm3697_data }, |
| 158 | { } |
| 159 | }; |
| 160 | MODULE_DEVICE_TABLE(of, ti_lmu_of_match); |
| 161 | |
| 162 | static int ti_lmu_probe(struct i2c_client *cl, const struct i2c_device_id *id) |
| 163 | { |
| 164 | struct device *dev = &cl->dev; |
| 165 | const struct of_device_id *match; |
| 166 | const struct ti_lmu_data *data; |
| 167 | struct regmap_config regmap_cfg; |
| 168 | struct ti_lmu *lmu; |
| 169 | int ret; |
| 170 | |
| 171 | match = of_match_device(ti_lmu_of_match, dev); |
| 172 | if (!match) |
| 173 | return -ENODEV; |
| 174 | /* |
| 175 | * Get device specific data from of_match table. |
| 176 | * This data is defined by using TI_LMU_DATA() macro. |
| 177 | */ |
| 178 | data = (struct ti_lmu_data *)match->data; |
| 179 | |
| 180 | lmu = devm_kzalloc(dev, sizeof(*lmu), GFP_KERNEL); |
| 181 | if (!lmu) |
| 182 | return -ENOMEM; |
| 183 | |
| 184 | lmu->dev = &cl->dev; |
| 185 | |
| 186 | /* Setup regmap */ |
| 187 | memset(®map_cfg, 0, sizeof(struct regmap_config)); |
| 188 | regmap_cfg.reg_bits = 8; |
| 189 | regmap_cfg.val_bits = 8; |
| 190 | regmap_cfg.name = id->name; |
| 191 | regmap_cfg.max_register = data->max_register; |
| 192 | |
| 193 | lmu->regmap = devm_regmap_init_i2c(cl, ®map_cfg); |
| 194 | if (IS_ERR(lmu->regmap)) |
| 195 | return PTR_ERR(lmu->regmap); |
| 196 | |
| 197 | /* HW enable pin control and additional power up sequence if required */ |
Pavel Machek | 7a6a395 | 2018-09-11 23:01:33 +0200 | [diff] [blame] | 198 | lmu->en_gpio = devm_gpiod_get_optional(dev, "enable", GPIOD_OUT_HIGH); |
| 199 | if (IS_ERR(lmu->en_gpio)) { |
| 200 | ret = PTR_ERR(lmu->en_gpio); |
| 201 | dev_err(dev, "Can not request enable GPIO: %d\n", ret); |
| 202 | return ret; |
| 203 | } |
| 204 | |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 205 | ret = ti_lmu_enable_hw(lmu, id->driver_data); |
| 206 | if (ret) |
| 207 | return ret; |
| 208 | |
Pavel Machek | 7891d37 | 2018-09-11 23:03:44 +0200 | [diff] [blame^] | 209 | ret = devm_add_action_or_reset(dev, ti_lmu_disable_hw, lmu); |
| 210 | if (ret) |
| 211 | return ret; |
| 212 | |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 213 | /* |
| 214 | * Fault circuit(open/short) can be detected by ti-lmu-fault-monitor. |
| 215 | * After fault detection is done, some devices should re-initialize |
| 216 | * configuration. The notifier enables such kind of handling. |
| 217 | */ |
| 218 | BLOCKING_INIT_NOTIFIER_HEAD(&lmu->notifier); |
| 219 | |
| 220 | i2c_set_clientdata(cl, lmu); |
| 221 | |
Pavel Machek | 7891d37 | 2018-09-11 23:03:44 +0200 | [diff] [blame^] | 222 | return devm_mfd_add_devices(lmu->dev, 0, data->cells, |
| 223 | data->num_cells, NULL, 0, NULL); |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 224 | } |
| 225 | |
| 226 | static const struct i2c_device_id ti_lmu_ids[] = { |
| 227 | { "lm3532", LM3532 }, |
| 228 | { "lm3631", LM3631 }, |
| 229 | { "lm3632", LM3632 }, |
| 230 | { "lm3633", LM3633 }, |
| 231 | { "lm3695", LM3695 }, |
| 232 | { "lm3697", LM3697 }, |
| 233 | { } |
| 234 | }; |
| 235 | MODULE_DEVICE_TABLE(i2c, ti_lmu_ids); |
| 236 | |
| 237 | static struct i2c_driver ti_lmu_driver = { |
| 238 | .probe = ti_lmu_probe, |
Milo Kim | d5aa11b | 2017-02-28 15:45:15 +0900 | [diff] [blame] | 239 | .driver = { |
| 240 | .name = "ti-lmu", |
| 241 | .of_match_table = ti_lmu_of_match, |
| 242 | }, |
| 243 | .id_table = ti_lmu_ids, |
| 244 | }; |
| 245 | |
| 246 | module_i2c_driver(ti_lmu_driver); |
| 247 | |
| 248 | MODULE_DESCRIPTION("TI LMU MFD Core Driver"); |
| 249 | MODULE_AUTHOR("Milo Kim"); |
| 250 | MODULE_LICENSE("GPL v2"); |