clk-vexpress.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. /*
  2. * This program is free software; you can redistribute it and/or modify
  3. * it under the terms of the GNU General Public License version 2 as
  4. * published by the Free Software Foundation.
  5. *
  6. * This program is distributed in the hope that it will be useful,
  7. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  8. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  9. * GNU General Public License for more details.
  10. *
  11. * Copyright (C) 2012 ARM Limited
  12. */
  13. #include <linux/clkdev.h>
  14. #include <linux/clk-provider.h>
  15. #include <linux/err.h>
  16. #include <linux/of.h>
  17. #include <linux/of_address.h>
  18. #include <linux/vexpress.h>
  19. #include <asm/hardware/sp810.h>
  20. static struct clk *vexpress_sp810_timerclken[4];
  21. static DEFINE_SPINLOCK(vexpress_sp810_lock);
  22. static void __init vexpress_sp810_init(void __iomem *base)
  23. {
  24. int i;
  25. if (WARN_ON(!base))
  26. return;
  27. for (i = 0; i < ARRAY_SIZE(vexpress_sp810_timerclken); i++) {
  28. char name[12];
  29. const char *parents[] = {
  30. "v2m:refclk32khz", /* REFCLK */
  31. "v2m:refclk1mhz" /* TIMCLK */
  32. };
  33. snprintf(name, ARRAY_SIZE(name), "timerclken%d", i);
  34. vexpress_sp810_timerclken[i] = clk_register_mux(NULL, name,
  35. parents, 2, 0, base + SCCTRL,
  36. SCCTRL_TIMERENnSEL_SHIFT(i), 1,
  37. 0, &vexpress_sp810_lock);
  38. if (WARN_ON(IS_ERR(vexpress_sp810_timerclken[i])))
  39. break;
  40. }
  41. }
  42. static const char * const vexpress_clk_24mhz_periphs[] __initconst = {
  43. "mb:uart0", "mb:uart1", "mb:uart2", "mb:uart3",
  44. "mb:mmci", "mb:kmi0", "mb:kmi1"
  45. };
  46. void __init vexpress_clk_init(void __iomem *sp810_base)
  47. {
  48. struct clk *clk;
  49. int i;
  50. clk = clk_register_fixed_rate(NULL, "dummy_apb_pclk", NULL,
  51. CLK_IS_ROOT, 0);
  52. WARN_ON(clk_register_clkdev(clk, "apb_pclk", NULL));
  53. clk = clk_register_fixed_rate(NULL, "v2m:clk_24mhz", NULL,
  54. CLK_IS_ROOT, 24000000);
  55. for (i = 0; i < ARRAY_SIZE(vexpress_clk_24mhz_periphs); i++)
  56. WARN_ON(clk_register_clkdev(clk, NULL,
  57. vexpress_clk_24mhz_periphs[i]));
  58. clk = clk_register_fixed_rate(NULL, "v2m:refclk32khz", NULL,
  59. CLK_IS_ROOT, 32768);
  60. WARN_ON(clk_register_clkdev(clk, NULL, "v2m:wdt"));
  61. clk = clk_register_fixed_rate(NULL, "v2m:refclk1mhz", NULL,
  62. CLK_IS_ROOT, 1000000);
  63. vexpress_sp810_init(sp810_base);
  64. for (i = 0; i < ARRAY_SIZE(vexpress_sp810_timerclken); i++)
  65. WARN_ON(clk_set_parent(vexpress_sp810_timerclken[i], clk));
  66. WARN_ON(clk_register_clkdev(vexpress_sp810_timerclken[0],
  67. "v2m-timer0", "sp804"));
  68. WARN_ON(clk_register_clkdev(vexpress_sp810_timerclken[1],
  69. "v2m-timer1", "sp804"));
  70. }
  71. #if defined(CONFIG_OF)
  72. struct clk *vexpress_sp810_of_get(struct of_phandle_args *clkspec, void *data)
  73. {
  74. if (WARN_ON(clkspec->args_count != 1 || clkspec->args[0] >
  75. ARRAY_SIZE(vexpress_sp810_timerclken)))
  76. return NULL;
  77. return vexpress_sp810_timerclken[clkspec->args[0]];
  78. }
  79. static const __initconst struct of_device_id vexpress_fixed_clk_match[] = {
  80. { .compatible = "fixed-clock", .data = of_fixed_clk_setup, },
  81. { .compatible = "arm,vexpress-osc", .data = vexpress_osc_of_setup, },
  82. {}
  83. };
  84. void __init vexpress_clk_of_init(void)
  85. {
  86. struct device_node *node;
  87. struct clk *clk;
  88. struct clk *refclk, *timclk;
  89. of_clk_init(vexpress_fixed_clk_match);
  90. node = of_find_compatible_node(NULL, NULL, "arm,sp810");
  91. vexpress_sp810_init(of_iomap(node, 0));
  92. of_clk_add_provider(node, vexpress_sp810_of_get, NULL);
  93. /* Select "better" (faster) parent for SP804 timers */
  94. refclk = of_clk_get_by_name(node, "refclk");
  95. timclk = of_clk_get_by_name(node, "timclk");
  96. if (!WARN_ON(IS_ERR(refclk) || IS_ERR(timclk))) {
  97. int i = 0;
  98. if (clk_get_rate(refclk) > clk_get_rate(timclk))
  99. clk = refclk;
  100. else
  101. clk = timclk;
  102. for (i = 0; i < ARRAY_SIZE(vexpress_sp810_timerclken); i++)
  103. WARN_ON(clk_set_parent(vexpress_sp810_timerclken[i],
  104. clk));
  105. }
  106. WARN_ON(clk_register_clkdev(vexpress_sp810_timerclken[0],
  107. "v2m-timer0", "sp804"));
  108. WARN_ON(clk_register_clkdev(vexpress_sp810_timerclken[1],
  109. "v2m-timer1", "sp804"));
  110. }
  111. #endif