clock-emev2.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. /*
  2. * Emma Mobile EV2 clock framework support
  3. *
  4. * Copyright (C) 2012 Magnus Damm
  5. *
  6. * This program is free software; you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License as published by
  8. * the Free Software Foundation; version 2 of the License.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program; if not, write to the Free Software
  17. * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
  18. */
  19. #include <linux/init.h>
  20. #include <linux/kernel.h>
  21. #include <linux/io.h>
  22. #include <linux/sh_clk.h>
  23. #include <linux/clkdev.h>
  24. #include <mach/common.h>
  25. #define EMEV2_SMU_BASE 0xe0110000
  26. /* EMEV2 SMU registers */
  27. #define USIAU0_RSTCTRL 0x094
  28. #define USIBU1_RSTCTRL 0x0ac
  29. #define USIBU2_RSTCTRL 0x0b0
  30. #define USIBU3_RSTCTRL 0x0b4
  31. #define STI_RSTCTRL 0x124
  32. #define USIAU0GCLKCTRL 0x4a0
  33. #define USIBU1GCLKCTRL 0x4b8
  34. #define USIBU2GCLKCTRL 0x4bc
  35. #define USIBU3GCLKCTRL 0x04c0
  36. #define STIGCLKCTRL 0x528
  37. #define USIAU0SCLKDIV 0x61c
  38. #define USIB2SCLKDIV 0x65c
  39. #define USIB3SCLKDIV 0x660
  40. #define STI_CLKSEL 0x688
  41. /* not pretty, but hey */
  42. static void __iomem *smu_base;
  43. static void emev2_smu_write(unsigned long value, int offs)
  44. {
  45. BUG_ON(!smu_base || (offs >= PAGE_SIZE));
  46. iowrite32(value, smu_base + offs);
  47. }
  48. static struct clk_mapping smu_mapping = {
  49. .phys = EMEV2_SMU_BASE,
  50. .len = PAGE_SIZE,
  51. };
  52. /* Fixed 32 KHz root clock from C32K pin */
  53. static struct clk c32k_clk = {
  54. .rate = 32768,
  55. .mapping = &smu_mapping,
  56. };
  57. /* PLL3 multiplies C32K with 7000 */
  58. static unsigned long pll3_recalc(struct clk *clk)
  59. {
  60. return clk->parent->rate * 7000;
  61. }
  62. static struct sh_clk_ops pll3_clk_ops = {
  63. .recalc = pll3_recalc,
  64. };
  65. static struct clk pll3_clk = {
  66. .ops = &pll3_clk_ops,
  67. .parent = &c32k_clk,
  68. };
  69. static struct clk *main_clks[] = {
  70. &c32k_clk,
  71. &pll3_clk,
  72. };
  73. enum { SCLKDIV_USIAU0, SCLKDIV_USIBU2, SCLKDIV_USIBU1, SCLKDIV_USIBU3,
  74. SCLKDIV_NR };
  75. #define SCLKDIV(_reg, _shift) \
  76. { \
  77. .parent = &pll3_clk, \
  78. .enable_reg = IOMEM(EMEV2_SMU_BASE + (_reg)), \
  79. .enable_bit = _shift, \
  80. }
  81. static struct clk sclkdiv_clks[SCLKDIV_NR] = {
  82. [SCLKDIV_USIAU0] = SCLKDIV(USIAU0SCLKDIV, 0),
  83. [SCLKDIV_USIBU2] = SCLKDIV(USIB2SCLKDIV, 16),
  84. [SCLKDIV_USIBU1] = SCLKDIV(USIB2SCLKDIV, 0),
  85. [SCLKDIV_USIBU3] = SCLKDIV(USIB3SCLKDIV, 0),
  86. };
  87. enum { GCLK_USIAU0_SCLK, GCLK_USIBU1_SCLK, GCLK_USIBU2_SCLK, GCLK_USIBU3_SCLK,
  88. GCLK_STI_SCLK,
  89. GCLK_NR };
  90. #define GCLK_SCLK(_parent, _reg) \
  91. { \
  92. .parent = _parent, \
  93. .enable_reg = IOMEM(EMEV2_SMU_BASE + (_reg)), \
  94. .enable_bit = 1, /* SCLK_GCC */ \
  95. }
  96. static struct clk gclk_clks[GCLK_NR] = {
  97. [GCLK_USIAU0_SCLK] = GCLK_SCLK(&sclkdiv_clks[SCLKDIV_USIAU0],
  98. USIAU0GCLKCTRL),
  99. [GCLK_USIBU1_SCLK] = GCLK_SCLK(&sclkdiv_clks[SCLKDIV_USIBU1],
  100. USIBU1GCLKCTRL),
  101. [GCLK_USIBU2_SCLK] = GCLK_SCLK(&sclkdiv_clks[SCLKDIV_USIBU2],
  102. USIBU2GCLKCTRL),
  103. [GCLK_USIBU3_SCLK] = GCLK_SCLK(&sclkdiv_clks[SCLKDIV_USIBU3],
  104. USIBU3GCLKCTRL),
  105. [GCLK_STI_SCLK] = GCLK_SCLK(&c32k_clk, STIGCLKCTRL),
  106. };
  107. static int emev2_gclk_enable(struct clk *clk)
  108. {
  109. iowrite32(ioread32(clk->mapped_reg) | (1 << clk->enable_bit),
  110. clk->mapped_reg);
  111. return 0;
  112. }
  113. static void emev2_gclk_disable(struct clk *clk)
  114. {
  115. iowrite32(ioread32(clk->mapped_reg) & ~(1 << clk->enable_bit),
  116. clk->mapped_reg);
  117. }
  118. static struct sh_clk_ops emev2_gclk_clk_ops = {
  119. .enable = emev2_gclk_enable,
  120. .disable = emev2_gclk_disable,
  121. .recalc = followparent_recalc,
  122. };
  123. static int __init emev2_gclk_register(struct clk *clks, int nr)
  124. {
  125. struct clk *clkp;
  126. int ret = 0;
  127. int k;
  128. for (k = 0; !ret && (k < nr); k++) {
  129. clkp = clks + k;
  130. clkp->ops = &emev2_gclk_clk_ops;
  131. ret |= clk_register(clkp);
  132. }
  133. return ret;
  134. }
  135. static unsigned long emev2_sclkdiv_recalc(struct clk *clk)
  136. {
  137. unsigned int sclk_div;
  138. sclk_div = (ioread32(clk->mapped_reg) >> clk->enable_bit) & 0xff;
  139. return clk->parent->rate / (sclk_div + 1);
  140. }
  141. static struct sh_clk_ops emev2_sclkdiv_clk_ops = {
  142. .recalc = emev2_sclkdiv_recalc,
  143. };
  144. static int __init emev2_sclkdiv_register(struct clk *clks, int nr)
  145. {
  146. struct clk *clkp;
  147. int ret = 0;
  148. int k;
  149. for (k = 0; !ret && (k < nr); k++) {
  150. clkp = clks + k;
  151. clkp->ops = &emev2_sclkdiv_clk_ops;
  152. ret |= clk_register(clkp);
  153. }
  154. return ret;
  155. }
  156. static struct clk_lookup lookups[] = {
  157. CLKDEV_DEV_ID("serial8250-em.0", &gclk_clks[GCLK_USIAU0_SCLK]),
  158. CLKDEV_DEV_ID("e1020000.uart", &gclk_clks[GCLK_USIAU0_SCLK]),
  159. CLKDEV_DEV_ID("serial8250-em.1", &gclk_clks[GCLK_USIBU1_SCLK]),
  160. CLKDEV_DEV_ID("e1030000.uart", &gclk_clks[GCLK_USIBU1_SCLK]),
  161. CLKDEV_DEV_ID("serial8250-em.2", &gclk_clks[GCLK_USIBU2_SCLK]),
  162. CLKDEV_DEV_ID("e1040000.uart", &gclk_clks[GCLK_USIBU2_SCLK]),
  163. CLKDEV_DEV_ID("serial8250-em.3", &gclk_clks[GCLK_USIBU3_SCLK]),
  164. CLKDEV_DEV_ID("e1050000.uart", &gclk_clks[GCLK_USIBU3_SCLK]),
  165. CLKDEV_DEV_ID("em_sti.0", &gclk_clks[GCLK_STI_SCLK]),
  166. CLKDEV_DEV_ID("e0180000.sti", &gclk_clks[GCLK_STI_SCLK]),
  167. };
  168. void __init emev2_clock_init(void)
  169. {
  170. int k, ret = 0;
  171. smu_base = ioremap(EMEV2_SMU_BASE, PAGE_SIZE);
  172. BUG_ON(!smu_base);
  173. /* setup STI timer to run on 32.768 kHz and deassert reset */
  174. emev2_smu_write(0, STI_CLKSEL);
  175. emev2_smu_write(1, STI_RSTCTRL);
  176. /* deassert reset for UART0->UART3 */
  177. emev2_smu_write(2, USIAU0_RSTCTRL);
  178. emev2_smu_write(2, USIBU1_RSTCTRL);
  179. emev2_smu_write(2, USIBU2_RSTCTRL);
  180. emev2_smu_write(2, USIBU3_RSTCTRL);
  181. for (k = 0; !ret && (k < ARRAY_SIZE(main_clks)); k++)
  182. ret = clk_register(main_clks[k]);
  183. if (!ret)
  184. ret = emev2_sclkdiv_register(sclkdiv_clks, SCLKDIV_NR);
  185. if (!ret)
  186. ret = emev2_gclk_register(gclk_clks, GCLK_NR);
  187. clkdev_add_table(lookups, ARRAY_SIZE(lookups));
  188. if (!ret)
  189. shmobile_clk_init();
  190. else
  191. panic("failed to setup emev2 clocks\n");
  192. }