jz4770-cgu.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * JZ4770 SoC CGU driver
  4. * Copyright 2018, Paul Cercueil <[email protected]>
  5. */
  6. #include <linux/bitops.h>
  7. #include <linux/clk-provider.h>
  8. #include <linux/delay.h>
  9. #include <linux/io.h>
  10. #include <linux/of.h>
  11. #include <dt-bindings/clock/ingenic,jz4770-cgu.h>
  12. #include "cgu.h"
  13. #include "pm.h"
  14. /*
  15. * CPM registers offset address definition
  16. */
  17. #define CGU_REG_CPCCR 0x00
  18. #define CGU_REG_LCR 0x04
  19. #define CGU_REG_CPPCR0 0x10
  20. #define CGU_REG_CLKGR0 0x20
  21. #define CGU_REG_OPCR 0x24
  22. #define CGU_REG_CLKGR1 0x28
  23. #define CGU_REG_CPPCR1 0x30
  24. #define CGU_REG_USBPCR1 0x48
  25. #define CGU_REG_USBCDR 0x50
  26. #define CGU_REG_I2SCDR 0x60
  27. #define CGU_REG_LPCDR 0x64
  28. #define CGU_REG_MSC0CDR 0x68
  29. #define CGU_REG_UHCCDR 0x6c
  30. #define CGU_REG_SSICDR 0x74
  31. #define CGU_REG_CIMCDR 0x7c
  32. #define CGU_REG_GPSCDR 0x80
  33. #define CGU_REG_PCMCDR 0x84
  34. #define CGU_REG_GPUCDR 0x88
  35. #define CGU_REG_MSC1CDR 0xA4
  36. #define CGU_REG_MSC2CDR 0xA8
  37. #define CGU_REG_BCHCDR 0xAC
  38. /* bits within the OPCR register */
  39. #define OPCR_SPENDH BIT(5) /* UHC PHY suspend */
  40. /* bits within the USBPCR1 register */
  41. #define USBPCR1_UHC_POWER BIT(5) /* UHC PHY power down */
  42. static struct ingenic_cgu *cgu;
  43. static int jz4770_uhc_phy_enable(struct clk_hw *hw)
  44. {
  45. void __iomem *reg_opcr = cgu->base + CGU_REG_OPCR;
  46. void __iomem *reg_usbpcr1 = cgu->base + CGU_REG_USBPCR1;
  47. writel(readl(reg_opcr) & ~OPCR_SPENDH, reg_opcr);
  48. writel(readl(reg_usbpcr1) | USBPCR1_UHC_POWER, reg_usbpcr1);
  49. return 0;
  50. }
  51. static void jz4770_uhc_phy_disable(struct clk_hw *hw)
  52. {
  53. void __iomem *reg_opcr = cgu->base + CGU_REG_OPCR;
  54. void __iomem *reg_usbpcr1 = cgu->base + CGU_REG_USBPCR1;
  55. writel(readl(reg_usbpcr1) & ~USBPCR1_UHC_POWER, reg_usbpcr1);
  56. writel(readl(reg_opcr) | OPCR_SPENDH, reg_opcr);
  57. }
  58. static int jz4770_uhc_phy_is_enabled(struct clk_hw *hw)
  59. {
  60. void __iomem *reg_opcr = cgu->base + CGU_REG_OPCR;
  61. void __iomem *reg_usbpcr1 = cgu->base + CGU_REG_USBPCR1;
  62. return !(readl(reg_opcr) & OPCR_SPENDH) &&
  63. (readl(reg_usbpcr1) & USBPCR1_UHC_POWER);
  64. }
  65. static const struct clk_ops jz4770_uhc_phy_ops = {
  66. .enable = jz4770_uhc_phy_enable,
  67. .disable = jz4770_uhc_phy_disable,
  68. .is_enabled = jz4770_uhc_phy_is_enabled,
  69. };
  70. static const s8 pll_od_encoding[8] = {
  71. 0x0, 0x1, -1, 0x2, -1, -1, -1, 0x3,
  72. };
  73. static const u8 jz4770_cgu_cpccr_div_table[] = {
  74. 1, 2, 3, 4, 6, 8, 12,
  75. };
  76. static const struct ingenic_cgu_clk_info jz4770_cgu_clocks[] = {
  77. /* External clocks */
  78. [JZ4770_CLK_EXT] = { "ext", CGU_CLK_EXT },
  79. [JZ4770_CLK_OSC32K] = { "osc32k", CGU_CLK_EXT },
  80. /* PLLs */
  81. [JZ4770_CLK_PLL0] = {
  82. "pll0", CGU_CLK_PLL,
  83. .parents = { JZ4770_CLK_EXT },
  84. .pll = {
  85. .reg = CGU_REG_CPPCR0,
  86. .rate_multiplier = 1,
  87. .m_shift = 24,
  88. .m_bits = 7,
  89. .m_offset = 1,
  90. .n_shift = 18,
  91. .n_bits = 5,
  92. .n_offset = 1,
  93. .od_shift = 16,
  94. .od_bits = 2,
  95. .od_max = 8,
  96. .od_encoding = pll_od_encoding,
  97. .bypass_reg = CGU_REG_CPPCR0,
  98. .bypass_bit = 9,
  99. .enable_bit = 8,
  100. .stable_bit = 10,
  101. },
  102. },
  103. [JZ4770_CLK_PLL1] = {
  104. /* TODO: PLL1 can depend on PLL0 */
  105. "pll1", CGU_CLK_PLL,
  106. .parents = { JZ4770_CLK_EXT },
  107. .pll = {
  108. .reg = CGU_REG_CPPCR1,
  109. .rate_multiplier = 1,
  110. .m_shift = 24,
  111. .m_bits = 7,
  112. .m_offset = 1,
  113. .n_shift = 18,
  114. .n_bits = 5,
  115. .n_offset = 1,
  116. .od_shift = 16,
  117. .od_bits = 2,
  118. .od_max = 8,
  119. .od_encoding = pll_od_encoding,
  120. .bypass_bit = -1,
  121. .enable_bit = 7,
  122. .stable_bit = 6,
  123. },
  124. },
  125. /* Main clocks */
  126. [JZ4770_CLK_CCLK] = {
  127. "cclk", CGU_CLK_DIV,
  128. /*
  129. * Disabling the CPU clock or any parent clocks will hang the
  130. * system; mark it critical.
  131. */
  132. .flags = CLK_IS_CRITICAL,
  133. .parents = { JZ4770_CLK_PLL0, },
  134. .div = {
  135. CGU_REG_CPCCR, 0, 1, 4, 22, -1, -1, 0,
  136. jz4770_cgu_cpccr_div_table,
  137. },
  138. },
  139. [JZ4770_CLK_H0CLK] = {
  140. "h0clk", CGU_CLK_DIV,
  141. .parents = { JZ4770_CLK_PLL0, },
  142. .div = {
  143. CGU_REG_CPCCR, 4, 1, 4, 22, -1, -1, 0,
  144. jz4770_cgu_cpccr_div_table,
  145. },
  146. },
  147. [JZ4770_CLK_H1CLK] = {
  148. "h1clk", CGU_CLK_DIV | CGU_CLK_GATE,
  149. .parents = { JZ4770_CLK_PLL0, },
  150. .div = {
  151. CGU_REG_CPCCR, 24, 1, 4, 22, -1, -1, 0,
  152. jz4770_cgu_cpccr_div_table,
  153. },
  154. .gate = { CGU_REG_CLKGR1, 7 },
  155. },
  156. [JZ4770_CLK_H2CLK] = {
  157. "h2clk", CGU_CLK_DIV,
  158. .parents = { JZ4770_CLK_PLL0, },
  159. .div = {
  160. CGU_REG_CPCCR, 16, 1, 4, 22, -1, -1, 0,
  161. jz4770_cgu_cpccr_div_table,
  162. },
  163. },
  164. [JZ4770_CLK_C1CLK] = {
  165. "c1clk", CGU_CLK_DIV | CGU_CLK_GATE,
  166. .parents = { JZ4770_CLK_PLL0, },
  167. .div = {
  168. CGU_REG_CPCCR, 12, 1, 4, 22, -1, -1, 0,
  169. jz4770_cgu_cpccr_div_table,
  170. },
  171. .gate = { CGU_REG_OPCR, 31, true }, // disable CCLK stop on idle
  172. },
  173. [JZ4770_CLK_PCLK] = {
  174. "pclk", CGU_CLK_DIV,
  175. .parents = { JZ4770_CLK_PLL0, },
  176. .div = {
  177. CGU_REG_CPCCR, 8, 1, 4, 22, -1, -1, 0,
  178. jz4770_cgu_cpccr_div_table,
  179. },
  180. },
  181. /* Those divided clocks can connect to PLL0 or PLL1 */
  182. [JZ4770_CLK_MMC0_MUX] = {
  183. "mmc0_mux", CGU_CLK_DIV | CGU_CLK_GATE | CGU_CLK_MUX,
  184. .parents = { JZ4770_CLK_PLL0, JZ4770_CLK_PLL1, },
  185. .mux = { CGU_REG_MSC0CDR, 30, 1 },
  186. .div = { CGU_REG_MSC0CDR, 0, 1, 7, -1, -1, 31 },
  187. .gate = { CGU_REG_MSC0CDR, 31 },
  188. },
  189. [JZ4770_CLK_MMC1_MUX] = {
  190. "mmc1_mux", CGU_CLK_DIV | CGU_CLK_GATE | CGU_CLK_MUX,
  191. .parents = { JZ4770_CLK_PLL0, JZ4770_CLK_PLL1, },
  192. .mux = { CGU_REG_MSC1CDR, 30, 1 },
  193. .div = { CGU_REG_MSC1CDR, 0, 1, 7, -1, -1, 31 },
  194. .gate = { CGU_REG_MSC1CDR, 31 },
  195. },
  196. [JZ4770_CLK_MMC2_MUX] = {
  197. "mmc2_mux", CGU_CLK_DIV | CGU_CLK_GATE | CGU_CLK_MUX,
  198. .parents = { JZ4770_CLK_PLL0, JZ4770_CLK_PLL1, },
  199. .mux = { CGU_REG_MSC2CDR, 30, 1 },
  200. .div = { CGU_REG_MSC2CDR, 0, 1, 7, -1, -1, 31 },
  201. .gate = { CGU_REG_MSC2CDR, 31 },
  202. },
  203. [JZ4770_CLK_CIM] = {
  204. "cim", CGU_CLK_DIV | CGU_CLK_GATE | CGU_CLK_MUX,
  205. .parents = { JZ4770_CLK_PLL0, JZ4770_CLK_PLL1, },
  206. .mux = { CGU_REG_CIMCDR, 31, 1 },
  207. .div = { CGU_REG_CIMCDR, 0, 1, 8, -1, -1, -1 },
  208. .gate = { CGU_REG_CLKGR0, 26 },
  209. },
  210. [JZ4770_CLK_UHC] = {
  211. "uhc", CGU_CLK_DIV | CGU_CLK_GATE | CGU_CLK_MUX,
  212. .parents = { JZ4770_CLK_PLL0, JZ4770_CLK_PLL1, },
  213. .mux = { CGU_REG_UHCCDR, 29, 1 },
  214. .div = { CGU_REG_UHCCDR, 0, 1, 4, -1, -1, -1 },
  215. .gate = { CGU_REG_CLKGR0, 24 },
  216. },
  217. [JZ4770_CLK_GPU] = {
  218. "gpu", CGU_CLK_DIV | CGU_CLK_GATE | CGU_CLK_MUX,
  219. .parents = { JZ4770_CLK_PLL0, JZ4770_CLK_PLL1, -1 },
  220. .mux = { CGU_REG_GPUCDR, 31, 1 },
  221. .div = { CGU_REG_GPUCDR, 0, 1, 3, -1, -1, -1 },
  222. .gate = { CGU_REG_CLKGR1, 9 },
  223. },
  224. [JZ4770_CLK_BCH] = {
  225. "bch", CGU_CLK_DIV | CGU_CLK_GATE | CGU_CLK_MUX,
  226. .parents = { JZ4770_CLK_PLL0, JZ4770_CLK_PLL1, },
  227. .mux = { CGU_REG_BCHCDR, 31, 1 },
  228. .div = { CGU_REG_BCHCDR, 0, 1, 3, -1, -1, -1 },
  229. .gate = { CGU_REG_CLKGR0, 1 },
  230. },
  231. [JZ4770_CLK_LPCLK_MUX] = {
  232. "lpclk", CGU_CLK_DIV | CGU_CLK_GATE | CGU_CLK_MUX,
  233. .parents = { JZ4770_CLK_PLL0, JZ4770_CLK_PLL1, },
  234. .mux = { CGU_REG_LPCDR, 29, 1 },
  235. .div = { CGU_REG_LPCDR, 0, 1, 11, -1, -1, -1 },
  236. .gate = { CGU_REG_CLKGR0, 28 },
  237. },
  238. [JZ4770_CLK_GPS] = {
  239. "gps", CGU_CLK_DIV | CGU_CLK_GATE | CGU_CLK_MUX,
  240. .parents = { JZ4770_CLK_PLL0, JZ4770_CLK_PLL1, },
  241. .mux = { CGU_REG_GPSCDR, 31, 1 },
  242. .div = { CGU_REG_GPSCDR, 0, 1, 4, -1, -1, -1 },
  243. .gate = { CGU_REG_CLKGR0, 22 },
  244. },
  245. /* Those divided clocks can connect to EXT, PLL0 or PLL1 */
  246. [JZ4770_CLK_SSI_MUX] = {
  247. "ssi_mux", CGU_CLK_DIV | CGU_CLK_MUX,
  248. .parents = { JZ4770_CLK_EXT, -1,
  249. JZ4770_CLK_PLL0, JZ4770_CLK_PLL1 },
  250. .mux = { CGU_REG_SSICDR, 30, 2 },
  251. .div = { CGU_REG_SSICDR, 0, 1, 6, -1, -1, -1 },
  252. },
  253. [JZ4770_CLK_PCM_MUX] = {
  254. "pcm_mux", CGU_CLK_DIV | CGU_CLK_MUX,
  255. .parents = { JZ4770_CLK_EXT, -1,
  256. JZ4770_CLK_PLL0, JZ4770_CLK_PLL1 },
  257. .mux = { CGU_REG_PCMCDR, 30, 2 },
  258. .div = { CGU_REG_PCMCDR, 0, 1, 9, -1, -1, -1 },
  259. },
  260. [JZ4770_CLK_I2S] = {
  261. "i2s", CGU_CLK_DIV | CGU_CLK_GATE | CGU_CLK_MUX,
  262. .parents = { JZ4770_CLK_EXT, -1,
  263. JZ4770_CLK_PLL0, JZ4770_CLK_PLL1 },
  264. .mux = { CGU_REG_I2SCDR, 30, 2 },
  265. .div = { CGU_REG_I2SCDR, 0, 1, 9, -1, -1, -1 },
  266. .gate = { CGU_REG_CLKGR1, 13 },
  267. },
  268. [JZ4770_CLK_OTG] = {
  269. "usb", CGU_CLK_DIV | CGU_CLK_GATE | CGU_CLK_MUX,
  270. .parents = { JZ4770_CLK_EXT, -1,
  271. JZ4770_CLK_PLL0, JZ4770_CLK_PLL1 },
  272. .mux = { CGU_REG_USBCDR, 30, 2 },
  273. .div = { CGU_REG_USBCDR, 0, 1, 8, -1, -1, -1 },
  274. .gate = { CGU_REG_CLKGR0, 2 },
  275. },
  276. /* Gate-only clocks */
  277. [JZ4770_CLK_SSI0] = {
  278. "ssi0", CGU_CLK_GATE,
  279. .parents = { JZ4770_CLK_SSI_MUX, },
  280. .gate = { CGU_REG_CLKGR0, 4 },
  281. },
  282. [JZ4770_CLK_SSI1] = {
  283. "ssi1", CGU_CLK_GATE,
  284. .parents = { JZ4770_CLK_SSI_MUX, },
  285. .gate = { CGU_REG_CLKGR0, 19 },
  286. },
  287. [JZ4770_CLK_SSI2] = {
  288. "ssi2", CGU_CLK_GATE,
  289. .parents = { JZ4770_CLK_SSI_MUX, },
  290. .gate = { CGU_REG_CLKGR0, 20 },
  291. },
  292. [JZ4770_CLK_PCM0] = {
  293. "pcm0", CGU_CLK_GATE,
  294. .parents = { JZ4770_CLK_PCM_MUX, },
  295. .gate = { CGU_REG_CLKGR1, 8 },
  296. },
  297. [JZ4770_CLK_PCM1] = {
  298. "pcm1", CGU_CLK_GATE,
  299. .parents = { JZ4770_CLK_PCM_MUX, },
  300. .gate = { CGU_REG_CLKGR1, 10 },
  301. },
  302. [JZ4770_CLK_DMA] = {
  303. "dma", CGU_CLK_GATE,
  304. .parents = { JZ4770_CLK_H2CLK, },
  305. .gate = { CGU_REG_CLKGR0, 21 },
  306. },
  307. [JZ4770_CLK_BDMA] = {
  308. "bdma", CGU_CLK_GATE,
  309. .parents = { JZ4770_CLK_H2CLK, },
  310. .gate = { CGU_REG_CLKGR1, 0 },
  311. },
  312. [JZ4770_CLK_I2C0] = {
  313. "i2c0", CGU_CLK_GATE,
  314. .parents = { JZ4770_CLK_EXT, },
  315. .gate = { CGU_REG_CLKGR0, 5 },
  316. },
  317. [JZ4770_CLK_I2C1] = {
  318. "i2c1", CGU_CLK_GATE,
  319. .parents = { JZ4770_CLK_EXT, },
  320. .gate = { CGU_REG_CLKGR0, 6 },
  321. },
  322. [JZ4770_CLK_I2C2] = {
  323. "i2c2", CGU_CLK_GATE,
  324. .parents = { JZ4770_CLK_EXT, },
  325. .gate = { CGU_REG_CLKGR1, 15 },
  326. },
  327. [JZ4770_CLK_UART0] = {
  328. "uart0", CGU_CLK_GATE,
  329. .parents = { JZ4770_CLK_EXT, },
  330. .gate = { CGU_REG_CLKGR0, 15 },
  331. },
  332. [JZ4770_CLK_UART1] = {
  333. "uart1", CGU_CLK_GATE,
  334. .parents = { JZ4770_CLK_EXT, },
  335. .gate = { CGU_REG_CLKGR0, 16 },
  336. },
  337. [JZ4770_CLK_UART2] = {
  338. "uart2", CGU_CLK_GATE,
  339. .parents = { JZ4770_CLK_EXT, },
  340. .gate = { CGU_REG_CLKGR0, 17 },
  341. },
  342. [JZ4770_CLK_UART3] = {
  343. "uart3", CGU_CLK_GATE,
  344. .parents = { JZ4770_CLK_EXT, },
  345. .gate = { CGU_REG_CLKGR0, 18 },
  346. },
  347. [JZ4770_CLK_IPU] = {
  348. "ipu", CGU_CLK_GATE,
  349. .parents = { JZ4770_CLK_H0CLK, },
  350. .gate = { CGU_REG_CLKGR0, 29 },
  351. },
  352. [JZ4770_CLK_ADC] = {
  353. "adc", CGU_CLK_GATE,
  354. .parents = { JZ4770_CLK_EXT, },
  355. .gate = { CGU_REG_CLKGR0, 14 },
  356. },
  357. [JZ4770_CLK_AIC] = {
  358. "aic", CGU_CLK_GATE,
  359. .parents = { JZ4770_CLK_EXT, },
  360. .gate = { CGU_REG_CLKGR0, 8 },
  361. },
  362. [JZ4770_CLK_AUX] = {
  363. "aux", CGU_CLK_GATE,
  364. .parents = { JZ4770_CLK_C1CLK, },
  365. .gate = { CGU_REG_CLKGR1, 14 },
  366. },
  367. [JZ4770_CLK_VPU] = {
  368. "vpu", CGU_CLK_GATE,
  369. .parents = { JZ4770_CLK_H1CLK, },
  370. .gate = { CGU_REG_LCR, 30, false, 150 },
  371. },
  372. [JZ4770_CLK_MMC0] = {
  373. "mmc0", CGU_CLK_GATE,
  374. .parents = { JZ4770_CLK_MMC0_MUX, },
  375. .gate = { CGU_REG_CLKGR0, 3 },
  376. },
  377. [JZ4770_CLK_MMC1] = {
  378. "mmc1", CGU_CLK_GATE,
  379. .parents = { JZ4770_CLK_MMC1_MUX, },
  380. .gate = { CGU_REG_CLKGR0, 11 },
  381. },
  382. [JZ4770_CLK_MMC2] = {
  383. "mmc2", CGU_CLK_GATE,
  384. .parents = { JZ4770_CLK_MMC2_MUX, },
  385. .gate = { CGU_REG_CLKGR0, 12 },
  386. },
  387. [JZ4770_CLK_OTG_PHY] = {
  388. "usb_phy", CGU_CLK_GATE,
  389. .parents = { JZ4770_CLK_OTG },
  390. .gate = { CGU_REG_OPCR, 7, true, 50 },
  391. },
  392. /* Custom clocks */
  393. [JZ4770_CLK_UHC_PHY] = {
  394. "uhc_phy", CGU_CLK_CUSTOM,
  395. .parents = { JZ4770_CLK_UHC, -1, -1, -1 },
  396. .custom = { &jz4770_uhc_phy_ops },
  397. },
  398. [JZ4770_CLK_EXT512] = {
  399. "ext/512", CGU_CLK_FIXDIV,
  400. .parents = { JZ4770_CLK_EXT },
  401. .fixdiv = { 512 },
  402. },
  403. [JZ4770_CLK_RTC] = {
  404. "rtc", CGU_CLK_MUX,
  405. .parents = { JZ4770_CLK_EXT512, JZ4770_CLK_OSC32K, },
  406. .mux = { CGU_REG_OPCR, 2, 1},
  407. },
  408. };
  409. static void __init jz4770_cgu_init(struct device_node *np)
  410. {
  411. int retval;
  412. cgu = ingenic_cgu_new(jz4770_cgu_clocks,
  413. ARRAY_SIZE(jz4770_cgu_clocks), np);
  414. if (!cgu) {
  415. pr_err("%s: failed to initialise CGU\n", __func__);
  416. return;
  417. }
  418. retval = ingenic_cgu_register_clocks(cgu);
  419. if (retval)
  420. pr_err("%s: failed to register CGU Clocks\n", __func__);
  421. ingenic_cgu_register_syscore_ops(cgu);
  422. }
  423. /* We only probe via devicetree, no need for a platform driver */
  424. CLK_OF_DECLARE_DRIVER(jz4770_cgu, "ingenic,jz4770-cgu", jz4770_cgu_init);