jz4770-cgu.c 12 KB

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