sy8827n.c 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. // SPDX-License-Identifier: GPL-2.0
  2. //
  3. // SY8827N regulator driver
  4. //
  5. // Copyright (C) 2020 Synaptics Incorporated
  6. //
  7. // Author: Jisheng Zhang <[email protected]>
  8. #include <linux/gpio/consumer.h>
  9. #include <linux/module.h>
  10. #include <linux/i2c.h>
  11. #include <linux/of_device.h>
  12. #include <linux/regmap.h>
  13. #include <linux/regulator/driver.h>
  14. #include <linux/regulator/of_regulator.h>
  15. #define SY8827N_VSEL0 0
  16. #define SY8827N_BUCK_EN (1 << 7)
  17. #define SY8827N_MODE (1 << 6)
  18. #define SY8827N_VSEL1 1
  19. #define SY8827N_CTRL 2
  20. #define SY8827N_ID1 3
  21. #define SY8827N_ID2 4
  22. #define SY8827N_PGOOD 5
  23. #define SY8827N_MAX (SY8827N_PGOOD + 1)
  24. #define SY8827N_NVOLTAGES 64
  25. #define SY8827N_VSELMIN 600000
  26. #define SY8827N_VSELSTEP 12500
  27. struct sy8827n_device_info {
  28. struct device *dev;
  29. struct regulator_desc desc;
  30. struct regulator_init_data *regulator;
  31. struct gpio_desc *en_gpio;
  32. unsigned int vsel_reg;
  33. };
  34. static int sy8827n_set_mode(struct regulator_dev *rdev, unsigned int mode)
  35. {
  36. struct sy8827n_device_info *di = rdev_get_drvdata(rdev);
  37. switch (mode) {
  38. case REGULATOR_MODE_FAST:
  39. regmap_update_bits(rdev->regmap, di->vsel_reg,
  40. SY8827N_MODE, SY8827N_MODE);
  41. break;
  42. case REGULATOR_MODE_NORMAL:
  43. regmap_update_bits(rdev->regmap, di->vsel_reg,
  44. SY8827N_MODE, 0);
  45. break;
  46. default:
  47. return -EINVAL;
  48. }
  49. return 0;
  50. }
  51. static unsigned int sy8827n_get_mode(struct regulator_dev *rdev)
  52. {
  53. struct sy8827n_device_info *di = rdev_get_drvdata(rdev);
  54. u32 val;
  55. int ret = 0;
  56. ret = regmap_read(rdev->regmap, di->vsel_reg, &val);
  57. if (ret < 0)
  58. return ret;
  59. if (val & SY8827N_MODE)
  60. return REGULATOR_MODE_FAST;
  61. else
  62. return REGULATOR_MODE_NORMAL;
  63. }
  64. static const struct regulator_ops sy8827n_regulator_ops = {
  65. .set_voltage_sel = regulator_set_voltage_sel_regmap,
  66. .get_voltage_sel = regulator_get_voltage_sel_regmap,
  67. .set_voltage_time_sel = regulator_set_voltage_time_sel,
  68. .map_voltage = regulator_map_voltage_linear,
  69. .list_voltage = regulator_list_voltage_linear,
  70. .enable = regulator_enable_regmap,
  71. .disable = regulator_disable_regmap,
  72. .is_enabled = regulator_is_enabled_regmap,
  73. .set_mode = sy8827n_set_mode,
  74. .get_mode = sy8827n_get_mode,
  75. };
  76. static int sy8827n_regulator_register(struct sy8827n_device_info *di,
  77. struct regulator_config *config)
  78. {
  79. struct regulator_desc *rdesc = &di->desc;
  80. struct regulator_dev *rdev;
  81. rdesc->name = "sy8827n-reg";
  82. rdesc->supply_name = "vin";
  83. rdesc->ops = &sy8827n_regulator_ops;
  84. rdesc->type = REGULATOR_VOLTAGE;
  85. rdesc->n_voltages = SY8827N_NVOLTAGES;
  86. rdesc->enable_reg = di->vsel_reg;
  87. rdesc->enable_mask = SY8827N_BUCK_EN;
  88. rdesc->min_uV = SY8827N_VSELMIN;
  89. rdesc->uV_step = SY8827N_VSELSTEP;
  90. rdesc->vsel_reg = di->vsel_reg;
  91. rdesc->vsel_mask = rdesc->n_voltages - 1;
  92. rdesc->owner = THIS_MODULE;
  93. rdev = devm_regulator_register(di->dev, &di->desc, config);
  94. return PTR_ERR_OR_ZERO(rdev);
  95. }
  96. static bool sy8827n_volatile_reg(struct device *dev, unsigned int reg)
  97. {
  98. if (reg == SY8827N_PGOOD)
  99. return true;
  100. return false;
  101. }
  102. static const struct regmap_config sy8827n_regmap_config = {
  103. .reg_bits = 8,
  104. .val_bits = 8,
  105. .volatile_reg = sy8827n_volatile_reg,
  106. .num_reg_defaults_raw = SY8827N_MAX,
  107. .cache_type = REGCACHE_FLAT,
  108. };
  109. static int sy8827n_i2c_probe(struct i2c_client *client)
  110. {
  111. struct device *dev = &client->dev;
  112. struct device_node *np = dev->of_node;
  113. struct sy8827n_device_info *di;
  114. struct regulator_config config = { };
  115. struct regmap *regmap;
  116. int ret;
  117. di = devm_kzalloc(dev, sizeof(struct sy8827n_device_info), GFP_KERNEL);
  118. if (!di)
  119. return -ENOMEM;
  120. di->regulator = of_get_regulator_init_data(dev, np, &di->desc);
  121. if (!di->regulator) {
  122. dev_err(dev, "Platform data not found!\n");
  123. return -EINVAL;
  124. }
  125. di->en_gpio = devm_gpiod_get_optional(dev, "enable", GPIOD_OUT_HIGH);
  126. if (IS_ERR(di->en_gpio))
  127. return PTR_ERR(di->en_gpio);
  128. if (of_property_read_bool(np, "silergy,vsel-state-high"))
  129. di->vsel_reg = SY8827N_VSEL1;
  130. else
  131. di->vsel_reg = SY8827N_VSEL0;
  132. di->dev = dev;
  133. regmap = devm_regmap_init_i2c(client, &sy8827n_regmap_config);
  134. if (IS_ERR(regmap)) {
  135. dev_err(dev, "Failed to allocate regmap!\n");
  136. return PTR_ERR(regmap);
  137. }
  138. i2c_set_clientdata(client, di);
  139. config.dev = di->dev;
  140. config.init_data = di->regulator;
  141. config.regmap = regmap;
  142. config.driver_data = di;
  143. config.of_node = np;
  144. ret = sy8827n_regulator_register(di, &config);
  145. if (ret < 0)
  146. dev_err(dev, "Failed to register regulator!\n");
  147. return ret;
  148. }
  149. #ifdef CONFIG_OF
  150. static const struct of_device_id sy8827n_dt_ids[] = {
  151. {
  152. .compatible = "silergy,sy8827n",
  153. },
  154. { }
  155. };
  156. MODULE_DEVICE_TABLE(of, sy8827n_dt_ids);
  157. #endif
  158. static const struct i2c_device_id sy8827n_id[] = {
  159. { "sy8827n", },
  160. { },
  161. };
  162. MODULE_DEVICE_TABLE(i2c, sy8827n_id);
  163. static struct i2c_driver sy8827n_regulator_driver = {
  164. .driver = {
  165. .name = "sy8827n-regulator",
  166. .of_match_table = of_match_ptr(sy8827n_dt_ids),
  167. },
  168. .probe_new = sy8827n_i2c_probe,
  169. .id_table = sy8827n_id,
  170. };
  171. module_i2c_driver(sy8827n_regulator_driver);
  172. MODULE_AUTHOR("Jisheng Zhang <[email protected]>");
  173. MODULE_DESCRIPTION("SY8827N regulator driver");
  174. MODULE_LICENSE("GPL v2");