clk-nspire.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. /*
  2. *
  3. * Copyright (C) 2013 Daniel Tang <tangrs@tangrs.id.au>
  4. *
  5. * This program is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License version 2, as
  7. * published by the Free Software Foundation.
  8. *
  9. */
  10. #include <linux/clk-provider.h>
  11. #include <linux/err.h>
  12. #include <linux/io.h>
  13. #include <linux/of.h>
  14. #include <linux/of_address.h>
  15. #define MHZ (1000 * 1000)
  16. #define BASE_CPU_SHIFT 1
  17. #define BASE_CPU_MASK 0x7F
  18. #define CPU_AHB_SHIFT 12
  19. #define CPU_AHB_MASK 0x07
  20. #define FIXED_BASE_SHIFT 8
  21. #define FIXED_BASE_MASK 0x01
  22. #define CLASSIC_BASE_SHIFT 16
  23. #define CLASSIC_BASE_MASK 0x1F
  24. #define CX_BASE_SHIFT 15
  25. #define CX_BASE_MASK 0x3F
  26. #define CX_UNKNOWN_SHIFT 21
  27. #define CX_UNKNOWN_MASK 0x03
  28. struct nspire_clk_info {
  29. u32 base_clock;
  30. u16 base_cpu_ratio;
  31. u16 base_ahb_ratio;
  32. };
  33. #define EXTRACT(var, prop) (((var)>>prop##_SHIFT) & prop##_MASK)
  34. static void nspire_clkinfo_cx(u32 val, struct nspire_clk_info *clk)
  35. {
  36. if (EXTRACT(val, FIXED_BASE))
  37. clk->base_clock = 48 * MHZ;
  38. else
  39. clk->base_clock = 6 * EXTRACT(val, CX_BASE) * MHZ;
  40. clk->base_cpu_ratio = EXTRACT(val, BASE_CPU) * EXTRACT(val, CX_UNKNOWN);
  41. clk->base_ahb_ratio = clk->base_cpu_ratio * (EXTRACT(val, CPU_AHB) + 1);
  42. }
  43. static void nspire_clkinfo_classic(u32 val, struct nspire_clk_info *clk)
  44. {
  45. if (EXTRACT(val, FIXED_BASE))
  46. clk->base_clock = 27 * MHZ;
  47. else
  48. clk->base_clock = (300 - 6 * EXTRACT(val, CLASSIC_BASE)) * MHZ;
  49. clk->base_cpu_ratio = EXTRACT(val, BASE_CPU) * 2;
  50. clk->base_ahb_ratio = clk->base_cpu_ratio * (EXTRACT(val, CPU_AHB) + 1);
  51. }
  52. static void __init nspire_ahbdiv_setup(struct device_node *node,
  53. void (*get_clkinfo)(u32, struct nspire_clk_info *))
  54. {
  55. u32 val;
  56. void __iomem *io;
  57. struct clk *clk;
  58. const char *clk_name = node->name;
  59. const char *parent_name;
  60. struct nspire_clk_info info;
  61. io = of_iomap(node, 0);
  62. if (!io)
  63. return;
  64. val = readl(io);
  65. iounmap(io);
  66. get_clkinfo(val, &info);
  67. of_property_read_string(node, "clock-output-names", &clk_name);
  68. parent_name = of_clk_get_parent_name(node, 0);
  69. clk = clk_register_fixed_factor(NULL, clk_name, parent_name, 0,
  70. 1, info.base_ahb_ratio);
  71. if (!IS_ERR(clk))
  72. of_clk_add_provider(node, of_clk_src_simple_get, clk);
  73. }
  74. static void __init nspire_ahbdiv_setup_cx(struct device_node *node)
  75. {
  76. nspire_ahbdiv_setup(node, nspire_clkinfo_cx);
  77. }
  78. static void __init nspire_ahbdiv_setup_classic(struct device_node *node)
  79. {
  80. nspire_ahbdiv_setup(node, nspire_clkinfo_classic);
  81. }
  82. CLK_OF_DECLARE(nspire_ahbdiv_cx, "lsi,nspire-cx-ahb-divider",
  83. nspire_ahbdiv_setup_cx);
  84. CLK_OF_DECLARE(nspire_ahbdiv_classic, "lsi,nspire-classic-ahb-divider",
  85. nspire_ahbdiv_setup_classic);
  86. static void __init nspire_clk_setup(struct device_node *node,
  87. void (*get_clkinfo)(u32, struct nspire_clk_info *))
  88. {
  89. u32 val;
  90. void __iomem *io;
  91. struct clk *clk;
  92. const char *clk_name = node->name;
  93. struct nspire_clk_info info;
  94. io = of_iomap(node, 0);
  95. if (!io)
  96. return;
  97. val = readl(io);
  98. iounmap(io);
  99. get_clkinfo(val, &info);
  100. of_property_read_string(node, "clock-output-names", &clk_name);
  101. clk = clk_register_fixed_rate(NULL, clk_name, NULL, CLK_IS_ROOT,
  102. info.base_clock);
  103. if (!IS_ERR(clk))
  104. of_clk_add_provider(node, of_clk_src_simple_get, clk);
  105. else
  106. return;
  107. pr_info("TI-NSPIRE Base: %uMHz CPU: %uMHz AHB: %uMHz\n",
  108. info.base_clock / MHZ,
  109. info.base_clock / info.base_cpu_ratio / MHZ,
  110. info.base_clock / info.base_ahb_ratio / MHZ);
  111. }
  112. static void __init nspire_clk_setup_cx(struct device_node *node)
  113. {
  114. nspire_clk_setup(node, nspire_clkinfo_cx);
  115. }
  116. static void __init nspire_clk_setup_classic(struct device_node *node)
  117. {
  118. nspire_clk_setup(node, nspire_clkinfo_classic);
  119. }
  120. CLK_OF_DECLARE(nspire_clk_cx, "lsi,nspire-cx-clock", nspire_clk_setup_cx);
  121. CLK_OF_DECLARE(nspire_clk_classic, "lsi,nspire-classic-clock",
  122. nspire_clk_setup_classic);