clk-twl.c 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Clock driver for twl device.
  4. *
  5. * inspired by the driver for the Palmas device
  6. */
  7. #include <linux/clk-provider.h>
  8. #include <linux/mfd/twl.h>
  9. #include <linux/module.h>
  10. #include <linux/platform_device.h>
  11. #include <linux/slab.h>
  12. #define VREG_STATE 2
  13. #define VREG_GRP 0
  14. #define TWL6030_CFG_STATE_OFF 0x00
  15. #define TWL6030_CFG_STATE_ON 0x01
  16. #define TWL6030_CFG_STATE_MASK 0x03
  17. #define TWL6030_CFG_STATE_GRP_SHIFT 5
  18. #define TWL6030_CFG_STATE_APP_SHIFT 2
  19. #define TWL6030_CFG_STATE_APP_MASK (0x03 << TWL6030_CFG_STATE_APP_SHIFT)
  20. #define TWL6030_CFG_STATE_APP(v) (((v) & TWL6030_CFG_STATE_APP_MASK) >>\
  21. TWL6030_CFG_STATE_APP_SHIFT)
  22. #define P1_GRP BIT(0) /* processor power group */
  23. #define P2_GRP BIT(1)
  24. #define P3_GRP BIT(2)
  25. #define ALL_GRP (P1_GRP | P2_GRP | P3_GRP)
  26. enum twl_type {
  27. TWL_TYPE_6030,
  28. TWL_TYPE_6032,
  29. };
  30. struct twl_clock_info {
  31. struct device *dev;
  32. enum twl_type type;
  33. u8 base;
  34. struct clk_hw hw;
  35. };
  36. static inline int
  37. twlclk_read(struct twl_clock_info *info, unsigned int slave_subgp,
  38. unsigned int offset)
  39. {
  40. u8 value;
  41. int status;
  42. status = twl_i2c_read_u8(slave_subgp, &value,
  43. info->base + offset);
  44. return (status < 0) ? status : value;
  45. }
  46. static inline int
  47. twlclk_write(struct twl_clock_info *info, unsigned int slave_subgp,
  48. unsigned int offset, u8 value)
  49. {
  50. return twl_i2c_write_u8(slave_subgp, value,
  51. info->base + offset);
  52. }
  53. static inline struct twl_clock_info *to_twl_clks_info(struct clk_hw *hw)
  54. {
  55. return container_of(hw, struct twl_clock_info, hw);
  56. }
  57. static unsigned long twl_clks_recalc_rate(struct clk_hw *hw,
  58. unsigned long parent_rate)
  59. {
  60. return 32768;
  61. }
  62. static int twl6032_clks_prepare(struct clk_hw *hw)
  63. {
  64. struct twl_clock_info *cinfo = to_twl_clks_info(hw);
  65. if (cinfo->type == TWL_TYPE_6030) {
  66. int grp;
  67. grp = twlclk_read(cinfo, TWL_MODULE_PM_RECEIVER, VREG_GRP);
  68. if (grp < 0)
  69. return grp;
  70. return twlclk_write(cinfo, TWL_MODULE_PM_RECEIVER, VREG_STATE,
  71. grp << TWL6030_CFG_STATE_GRP_SHIFT |
  72. TWL6030_CFG_STATE_ON);
  73. }
  74. return twlclk_write(cinfo, TWL_MODULE_PM_RECEIVER, VREG_STATE,
  75. TWL6030_CFG_STATE_ON);
  76. }
  77. static void twl6032_clks_unprepare(struct clk_hw *hw)
  78. {
  79. struct twl_clock_info *cinfo = to_twl_clks_info(hw);
  80. int ret;
  81. if (cinfo->type == TWL_TYPE_6030)
  82. ret = twlclk_write(cinfo, TWL_MODULE_PM_RECEIVER, VREG_STATE,
  83. ALL_GRP << TWL6030_CFG_STATE_GRP_SHIFT |
  84. TWL6030_CFG_STATE_OFF);
  85. else
  86. ret = twlclk_write(cinfo, TWL_MODULE_PM_RECEIVER, VREG_STATE,
  87. TWL6030_CFG_STATE_OFF);
  88. if (ret < 0)
  89. dev_err(cinfo->dev, "clk unprepare failed\n");
  90. }
  91. static const struct clk_ops twl6032_clks_ops = {
  92. .prepare = twl6032_clks_prepare,
  93. .unprepare = twl6032_clks_unprepare,
  94. .recalc_rate = twl_clks_recalc_rate,
  95. };
  96. struct twl_clks_data {
  97. struct clk_init_data init;
  98. u8 base;
  99. };
  100. static const struct twl_clks_data twl6032_clks[] = {
  101. {
  102. .init = {
  103. .name = "clk32kg",
  104. .ops = &twl6032_clks_ops,
  105. .flags = CLK_IGNORE_UNUSED,
  106. },
  107. .base = 0x8C,
  108. },
  109. {
  110. .init = {
  111. .name = "clk32kaudio",
  112. .ops = &twl6032_clks_ops,
  113. .flags = CLK_IGNORE_UNUSED,
  114. },
  115. .base = 0x8F,
  116. },
  117. {
  118. /* sentinel */
  119. }
  120. };
  121. static int twl_clks_probe(struct platform_device *pdev)
  122. {
  123. struct clk_hw_onecell_data *clk_data;
  124. const struct twl_clks_data *hw_data;
  125. struct twl_clock_info *cinfo;
  126. int ret;
  127. int i;
  128. int count;
  129. hw_data = twl6032_clks;
  130. for (count = 0; hw_data[count].init.name; count++)
  131. ;
  132. clk_data = devm_kzalloc(&pdev->dev,
  133. struct_size(clk_data, hws, count),
  134. GFP_KERNEL);
  135. if (!clk_data)
  136. return -ENOMEM;
  137. clk_data->num = count;
  138. cinfo = devm_kcalloc(&pdev->dev, count, sizeof(*cinfo), GFP_KERNEL);
  139. if (!cinfo)
  140. return -ENOMEM;
  141. for (i = 0; i < count; i++) {
  142. cinfo[i].base = hw_data[i].base;
  143. cinfo[i].dev = &pdev->dev;
  144. cinfo[i].type = platform_get_device_id(pdev)->driver_data;
  145. cinfo[i].hw.init = &hw_data[i].init;
  146. ret = devm_clk_hw_register(&pdev->dev, &cinfo[i].hw);
  147. if (ret) {
  148. return dev_err_probe(&pdev->dev, ret,
  149. "Fail to register clock %s\n",
  150. hw_data[i].init.name);
  151. }
  152. clk_data->hws[i] = &cinfo[i].hw;
  153. }
  154. ret = devm_of_clk_add_hw_provider(&pdev->dev,
  155. of_clk_hw_onecell_get, clk_data);
  156. if (ret < 0)
  157. return dev_err_probe(&pdev->dev, ret,
  158. "Fail to add clock driver\n");
  159. return 0;
  160. }
  161. static const struct platform_device_id twl_clks_id[] = {
  162. {
  163. .name = "twl6030-clk",
  164. .driver_data = TWL_TYPE_6030,
  165. }, {
  166. .name = "twl6032-clk",
  167. .driver_data = TWL_TYPE_6032,
  168. }, {
  169. /* sentinel */
  170. }
  171. };
  172. MODULE_DEVICE_TABLE(platform, twl_clks_id);
  173. static struct platform_driver twl_clks_driver = {
  174. .driver = {
  175. .name = "twl-clk",
  176. },
  177. .probe = twl_clks_probe,
  178. .id_table = twl_clks_id,
  179. };
  180. module_platform_driver(twl_clks_driver);
  181. MODULE_DESCRIPTION("Clock driver for TWL Series Devices");
  182. MODULE_LICENSE("GPL");