rcar-rst.c 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * R-Car Gen1 RESET/WDT, R-Car Gen2, Gen3, and RZ/G RST Driver
  4. *
  5. * Copyright (C) 2016 Glider bvba
  6. */
  7. #include <linux/err.h>
  8. #include <linux/io.h>
  9. #include <linux/of_address.h>
  10. #include <linux/soc/renesas/rcar-rst.h>
  11. #define WDTRSTCR_RESET 0xA55A0002
  12. #define WDTRSTCR 0x0054
  13. #define GEN4_WDTRSTCR 0x0010
  14. #define CR7BAR 0x0070
  15. #define CR7BAREN BIT(4)
  16. #define CR7BAR_MASK 0xFFFC0000
  17. static void __iomem *rcar_rst_base;
  18. static u32 saved_mode __initdata;
  19. static int (*rcar_rst_set_rproc_boot_addr_func)(u64 boot_addr);
  20. static int rcar_rst_enable_wdt_reset(void __iomem *base)
  21. {
  22. iowrite32(WDTRSTCR_RESET, base + WDTRSTCR);
  23. return 0;
  24. }
  25. static int rcar_rst_v3u_enable_wdt_reset(void __iomem *base)
  26. {
  27. iowrite32(WDTRSTCR_RESET, base + GEN4_WDTRSTCR);
  28. return 0;
  29. }
  30. /*
  31. * Most of the R-Car Gen3 SoCs have an ARM Realtime Core.
  32. * Firmware boot address has to be set in CR7BAR before
  33. * starting the realtime core.
  34. * Boot address must be aligned on a 256k boundary.
  35. */
  36. static int rcar_rst_set_gen3_rproc_boot_addr(u64 boot_addr)
  37. {
  38. if (boot_addr & ~(u64)CR7BAR_MASK) {
  39. pr_err("Invalid boot address got %llx\n", boot_addr);
  40. return -EINVAL;
  41. }
  42. iowrite32(boot_addr, rcar_rst_base + CR7BAR);
  43. iowrite32(boot_addr | CR7BAREN, rcar_rst_base + CR7BAR);
  44. return 0;
  45. }
  46. struct rst_config {
  47. unsigned int modemr; /* Mode Monitoring Register Offset */
  48. int (*configure)(void __iomem *base); /* Platform specific config */
  49. int (*set_rproc_boot_addr)(u64 boot_addr);
  50. };
  51. static const struct rst_config rcar_rst_gen1 __initconst = {
  52. .modemr = 0x20,
  53. };
  54. static const struct rst_config rcar_rst_gen2 __initconst = {
  55. .modemr = 0x60,
  56. .configure = rcar_rst_enable_wdt_reset,
  57. };
  58. static const struct rst_config rcar_rst_gen3 __initconst = {
  59. .modemr = 0x60,
  60. .set_rproc_boot_addr = rcar_rst_set_gen3_rproc_boot_addr,
  61. };
  62. /* V3U firmware doesn't enable WDT reset and there won't be updates anymore */
  63. static const struct rst_config rcar_rst_v3u __initconst = {
  64. .modemr = 0x00, /* MODEMR0 and it has CPG related bits */
  65. .configure = rcar_rst_v3u_enable_wdt_reset,
  66. };
  67. static const struct rst_config rcar_rst_gen4 __initconst = {
  68. .modemr = 0x00, /* MODEMR0 and it has CPG related bits */
  69. };
  70. static const struct of_device_id rcar_rst_matches[] __initconst = {
  71. /* RZ/G1 is handled like R-Car Gen2 */
  72. { .compatible = "renesas,r8a7742-rst", .data = &rcar_rst_gen2 },
  73. { .compatible = "renesas,r8a7743-rst", .data = &rcar_rst_gen2 },
  74. { .compatible = "renesas,r8a7744-rst", .data = &rcar_rst_gen2 },
  75. { .compatible = "renesas,r8a7745-rst", .data = &rcar_rst_gen2 },
  76. { .compatible = "renesas,r8a77470-rst", .data = &rcar_rst_gen2 },
  77. /* RZ/G2 is handled like R-Car Gen3 */
  78. { .compatible = "renesas,r8a774a1-rst", .data = &rcar_rst_gen3 },
  79. { .compatible = "renesas,r8a774b1-rst", .data = &rcar_rst_gen3 },
  80. { .compatible = "renesas,r8a774c0-rst", .data = &rcar_rst_gen3 },
  81. { .compatible = "renesas,r8a774e1-rst", .data = &rcar_rst_gen3 },
  82. /* R-Car Gen1 */
  83. { .compatible = "renesas,r8a7778-reset-wdt", .data = &rcar_rst_gen1 },
  84. { .compatible = "renesas,r8a7779-reset-wdt", .data = &rcar_rst_gen1 },
  85. /* R-Car Gen2 */
  86. { .compatible = "renesas,r8a7790-rst", .data = &rcar_rst_gen2 },
  87. { .compatible = "renesas,r8a7791-rst", .data = &rcar_rst_gen2 },
  88. { .compatible = "renesas,r8a7792-rst", .data = &rcar_rst_gen2 },
  89. { .compatible = "renesas,r8a7793-rst", .data = &rcar_rst_gen2 },
  90. { .compatible = "renesas,r8a7794-rst", .data = &rcar_rst_gen2 },
  91. /* R-Car Gen3 */
  92. { .compatible = "renesas,r8a7795-rst", .data = &rcar_rst_gen3 },
  93. { .compatible = "renesas,r8a7796-rst", .data = &rcar_rst_gen3 },
  94. { .compatible = "renesas,r8a77961-rst", .data = &rcar_rst_gen3 },
  95. { .compatible = "renesas,r8a77965-rst", .data = &rcar_rst_gen3 },
  96. { .compatible = "renesas,r8a77970-rst", .data = &rcar_rst_gen3 },
  97. { .compatible = "renesas,r8a77980-rst", .data = &rcar_rst_gen3 },
  98. { .compatible = "renesas,r8a77990-rst", .data = &rcar_rst_gen3 },
  99. { .compatible = "renesas,r8a77995-rst", .data = &rcar_rst_gen3 },
  100. /* R-Car Gen4 */
  101. { .compatible = "renesas,r8a779a0-rst", .data = &rcar_rst_v3u },
  102. { .compatible = "renesas,r8a779f0-rst", .data = &rcar_rst_gen4 },
  103. { .compatible = "renesas,r8a779g0-rst", .data = &rcar_rst_gen4 },
  104. { .compatible = "renesas,r8a779h0-rst", .data = &rcar_rst_gen4 },
  105. { /* sentinel */ }
  106. };
  107. static int __init rcar_rst_init(void)
  108. {
  109. const struct of_device_id *match;
  110. const struct rst_config *cfg;
  111. struct device_node *np;
  112. void __iomem *base;
  113. int error = 0;
  114. np = of_find_matching_node_and_match(NULL, rcar_rst_matches, &match);
  115. if (!np)
  116. return -ENODEV;
  117. base = of_iomap(np, 0);
  118. if (!base) {
  119. pr_warn("%pOF: Cannot map regs\n", np);
  120. error = -ENOMEM;
  121. goto out_put;
  122. }
  123. rcar_rst_base = base;
  124. cfg = match->data;
  125. rcar_rst_set_rproc_boot_addr_func = cfg->set_rproc_boot_addr;
  126. saved_mode = ioread32(base + cfg->modemr);
  127. if (cfg->configure) {
  128. error = cfg->configure(base);
  129. if (error) {
  130. pr_warn("%pOF: Cannot run SoC specific configuration\n",
  131. np);
  132. goto out_put;
  133. }
  134. }
  135. pr_debug("%pOF: MODE = 0x%08x\n", np, saved_mode);
  136. out_put:
  137. of_node_put(np);
  138. return error;
  139. }
  140. int __init rcar_rst_read_mode_pins(u32 *mode)
  141. {
  142. int error;
  143. if (!rcar_rst_base) {
  144. error = rcar_rst_init();
  145. if (error)
  146. return error;
  147. }
  148. *mode = saved_mode;
  149. return 0;
  150. }
  151. int rcar_rst_set_rproc_boot_addr(u64 boot_addr)
  152. {
  153. if (!rcar_rst_set_rproc_boot_addr_func)
  154. return -EIO;
  155. return rcar_rst_set_rproc_boot_addr_func(boot_addr);
  156. }
  157. EXPORT_SYMBOL_GPL(rcar_rst_set_rproc_boot_addr);