cros-ec-regulator.c 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. //
  3. // Copyright 2020 Google LLC.
  4. #include <linux/module.h>
  5. #include <linux/of.h>
  6. #include <linux/platform_data/cros_ec_proto.h>
  7. #include <linux/platform_device.h>
  8. #include <linux/regulator/driver.h>
  9. #include <linux/regulator/machine.h>
  10. #include <linux/regulator/of_regulator.h>
  11. #include <linux/slab.h>
  12. struct cros_ec_regulator_data {
  13. struct regulator_desc desc;
  14. struct regulator_dev *dev;
  15. struct cros_ec_device *ec_dev;
  16. u32 index;
  17. u16 *voltages_mV;
  18. u16 num_voltages;
  19. };
  20. static int cros_ec_regulator_enable(struct regulator_dev *dev)
  21. {
  22. struct cros_ec_regulator_data *data = rdev_get_drvdata(dev);
  23. struct ec_params_regulator_enable cmd = {
  24. .index = data->index,
  25. .enable = 1,
  26. };
  27. return cros_ec_cmd(data->ec_dev, 0, EC_CMD_REGULATOR_ENABLE, &cmd,
  28. sizeof(cmd), NULL, 0);
  29. }
  30. static int cros_ec_regulator_disable(struct regulator_dev *dev)
  31. {
  32. struct cros_ec_regulator_data *data = rdev_get_drvdata(dev);
  33. struct ec_params_regulator_enable cmd = {
  34. .index = data->index,
  35. .enable = 0,
  36. };
  37. return cros_ec_cmd(data->ec_dev, 0, EC_CMD_REGULATOR_ENABLE, &cmd,
  38. sizeof(cmd), NULL, 0);
  39. }
  40. static int cros_ec_regulator_is_enabled(struct regulator_dev *dev)
  41. {
  42. struct cros_ec_regulator_data *data = rdev_get_drvdata(dev);
  43. struct ec_params_regulator_is_enabled cmd = {
  44. .index = data->index,
  45. };
  46. struct ec_response_regulator_is_enabled resp;
  47. int ret;
  48. ret = cros_ec_cmd(data->ec_dev, 0, EC_CMD_REGULATOR_IS_ENABLED, &cmd,
  49. sizeof(cmd), &resp, sizeof(resp));
  50. if (ret < 0)
  51. return ret;
  52. return resp.enabled;
  53. }
  54. static int cros_ec_regulator_list_voltage(struct regulator_dev *dev,
  55. unsigned int selector)
  56. {
  57. struct cros_ec_regulator_data *data = rdev_get_drvdata(dev);
  58. if (selector >= data->num_voltages)
  59. return -EINVAL;
  60. return data->voltages_mV[selector] * 1000;
  61. }
  62. static int cros_ec_regulator_get_voltage(struct regulator_dev *dev)
  63. {
  64. struct cros_ec_regulator_data *data = rdev_get_drvdata(dev);
  65. struct ec_params_regulator_get_voltage cmd = {
  66. .index = data->index,
  67. };
  68. struct ec_response_regulator_get_voltage resp;
  69. int ret;
  70. ret = cros_ec_cmd(data->ec_dev, 0, EC_CMD_REGULATOR_GET_VOLTAGE, &cmd,
  71. sizeof(cmd), &resp, sizeof(resp));
  72. if (ret < 0)
  73. return ret;
  74. return resp.voltage_mv * 1000;
  75. }
  76. static int cros_ec_regulator_set_voltage(struct regulator_dev *dev, int min_uV,
  77. int max_uV, unsigned int *selector)
  78. {
  79. struct cros_ec_regulator_data *data = rdev_get_drvdata(dev);
  80. int min_mV = DIV_ROUND_UP(min_uV, 1000);
  81. int max_mV = max_uV / 1000;
  82. struct ec_params_regulator_set_voltage cmd = {
  83. .index = data->index,
  84. .min_mv = min_mV,
  85. .max_mv = max_mV,
  86. };
  87. /*
  88. * This can happen when the given range [min_uV, max_uV] doesn't
  89. * contain any voltage that can be represented exactly in mV.
  90. */
  91. if (min_mV > max_mV)
  92. return -EINVAL;
  93. return cros_ec_cmd(data->ec_dev, 0, EC_CMD_REGULATOR_SET_VOLTAGE, &cmd,
  94. sizeof(cmd), NULL, 0);
  95. }
  96. static const struct regulator_ops cros_ec_regulator_voltage_ops = {
  97. .enable = cros_ec_regulator_enable,
  98. .disable = cros_ec_regulator_disable,
  99. .is_enabled = cros_ec_regulator_is_enabled,
  100. .list_voltage = cros_ec_regulator_list_voltage,
  101. .get_voltage = cros_ec_regulator_get_voltage,
  102. .set_voltage = cros_ec_regulator_set_voltage,
  103. };
  104. static int cros_ec_regulator_init_info(struct device *dev,
  105. struct cros_ec_regulator_data *data)
  106. {
  107. struct ec_params_regulator_get_info cmd = {
  108. .index = data->index,
  109. };
  110. struct ec_response_regulator_get_info resp;
  111. int ret;
  112. ret = cros_ec_cmd(data->ec_dev, 0, EC_CMD_REGULATOR_GET_INFO, &cmd,
  113. sizeof(cmd), &resp, sizeof(resp));
  114. if (ret < 0)
  115. return ret;
  116. data->num_voltages =
  117. min_t(u16, ARRAY_SIZE(resp.voltages_mv), resp.num_voltages);
  118. data->voltages_mV =
  119. devm_kmemdup(dev, resp.voltages_mv,
  120. sizeof(u16) * data->num_voltages, GFP_KERNEL);
  121. if (!data->voltages_mV)
  122. return -ENOMEM;
  123. data->desc.n_voltages = data->num_voltages;
  124. /* Make sure the returned name is always a valid string */
  125. resp.name[ARRAY_SIZE(resp.name) - 1] = '\0';
  126. data->desc.name = devm_kstrdup(dev, resp.name, GFP_KERNEL);
  127. if (!data->desc.name)
  128. return -ENOMEM;
  129. return 0;
  130. }
  131. static int cros_ec_regulator_probe(struct platform_device *pdev)
  132. {
  133. struct device *dev = &pdev->dev;
  134. struct device_node *np = dev->of_node;
  135. struct cros_ec_regulator_data *drvdata;
  136. struct regulator_init_data *init_data;
  137. struct regulator_config cfg = {};
  138. struct regulator_desc *desc;
  139. int ret;
  140. drvdata = devm_kzalloc(
  141. &pdev->dev, sizeof(struct cros_ec_regulator_data), GFP_KERNEL);
  142. if (!drvdata)
  143. return -ENOMEM;
  144. drvdata->ec_dev = dev_get_drvdata(dev->parent);
  145. desc = &drvdata->desc;
  146. init_data = of_get_regulator_init_data(dev, np, desc);
  147. if (!init_data)
  148. return -EINVAL;
  149. ret = of_property_read_u32(np, "reg", &drvdata->index);
  150. if (ret < 0)
  151. return ret;
  152. desc->owner = THIS_MODULE;
  153. desc->type = REGULATOR_VOLTAGE;
  154. desc->ops = &cros_ec_regulator_voltage_ops;
  155. ret = cros_ec_regulator_init_info(dev, drvdata);
  156. if (ret < 0)
  157. return ret;
  158. cfg.dev = &pdev->dev;
  159. cfg.init_data = init_data;
  160. cfg.driver_data = drvdata;
  161. cfg.of_node = np;
  162. drvdata->dev = devm_regulator_register(dev, &drvdata->desc, &cfg);
  163. if (IS_ERR(drvdata->dev)) {
  164. ret = PTR_ERR(drvdata->dev);
  165. dev_err(&pdev->dev, "Failed to register regulator: %d\n", ret);
  166. return ret;
  167. }
  168. platform_set_drvdata(pdev, drvdata);
  169. return 0;
  170. }
  171. static const struct of_device_id regulator_cros_ec_of_match[] = {
  172. { .compatible = "google,cros-ec-regulator", },
  173. {}
  174. };
  175. MODULE_DEVICE_TABLE(of, regulator_cros_ec_of_match);
  176. static struct platform_driver cros_ec_regulator_driver = {
  177. .probe = cros_ec_regulator_probe,
  178. .driver = {
  179. .name = "cros-ec-regulator",
  180. .of_match_table = regulator_cros_ec_of_match,
  181. },
  182. };
  183. module_platform_driver(cros_ec_regulator_driver);
  184. MODULE_LICENSE("GPL v2");
  185. MODULE_DESCRIPTION("ChromeOS EC controlled regulator");
  186. MODULE_AUTHOR("Pi-Hsun Shih <[email protected]>");