lowlevel_init.c 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. /*
  2. * Lowlevel setup for EXYNOS5 based board
  3. *
  4. * Copyright (C) 2013 Samsung Electronics
  5. * Rajeshwari Shinde <rajeshwari.s@samsung.com>
  6. *
  7. * See file CREDITS for list of people who contributed to this
  8. * project.
  9. *
  10. * This program is free software; you can redistribute it and/or
  11. * modify it under the terms of the GNU General Public License as
  12. * published by the Free Software Foundation; either version 2 of
  13. * the License, or (at your option) any later version.
  14. *
  15. * This program is distributed in the hope that it will be useful,
  16. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. * GNU General Public License for more details.
  19. *
  20. * You should have received a copy of the GNU General Public License
  21. * along with this program; if not, write to the Free Software
  22. * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
  23. * MA 02111-1307 USA
  24. */
  25. #include <common.h>
  26. #include <config.h>
  27. #include <debug_uart.h>
  28. #include <asm/arch/cpu.h>
  29. #include <asm/arch/dmc.h>
  30. #include <asm/arch/power.h>
  31. #include <asm/arch/tzpc.h>
  32. #include <asm/arch/periph.h>
  33. #include <asm/arch/pinmux.h>
  34. #include <asm/arch/system.h>
  35. #include <asm/armv7.h>
  36. #include "common_setup.h"
  37. #include "exynos5_setup.h"
  38. /* These are the things we can do during low-level init */
  39. enum {
  40. DO_WAKEUP = 1 << 0,
  41. DO_CLOCKS = 1 << 1,
  42. DO_MEM_RESET = 1 << 2,
  43. DO_UART = 1 << 3,
  44. DO_POWER = 1 << 4,
  45. };
  46. #ifdef CONFIG_EXYNOS5420
  47. /*
  48. * Power up secondary CPUs.
  49. */
  50. static void secondary_cpu_start(void)
  51. {
  52. v7_enable_smp(EXYNOS5420_INFORM_BASE);
  53. svc32_mode_en();
  54. branch_bx(CONFIG_EXYNOS_RELOCATE_CODE_BASE);
  55. }
  56. /*
  57. * This is the entry point of hotplug-in and
  58. * cluster switching.
  59. */
  60. static void low_power_start(void)
  61. {
  62. uint32_t val, reg_val;
  63. reg_val = readl(EXYNOS5420_SPARE_BASE);
  64. if (reg_val != CPU_RST_FLAG_VAL) {
  65. writel(0x0, CONFIG_LOWPOWER_FLAG);
  66. branch_bx(0x0);
  67. }
  68. reg_val = readl(CONFIG_PHY_IRAM_BASE + 0x4);
  69. if (reg_val != (uint32_t)&low_power_start) {
  70. /* Store jump address as low_power_start if not present */
  71. writel((uint32_t)&low_power_start, CONFIG_PHY_IRAM_BASE + 0x4);
  72. dsb();
  73. sev();
  74. }
  75. /* Set the CPU to SVC32 mode */
  76. svc32_mode_en();
  77. #ifndef CONFIG_SYS_L2CACHE_OFF
  78. /* Read MIDR for Primary Part Number */
  79. mrc_midr(val);
  80. val = (val >> 4);
  81. val &= 0xf;
  82. if (val == 0xf) {
  83. configure_l2_ctlr();
  84. configure_l2_actlr();
  85. v7_enable_l2_hazard_detect();
  86. }
  87. #endif
  88. /* Invalidate L1 & TLB */
  89. val = 0x0;
  90. mcr_tlb(val);
  91. mcr_icache(val);
  92. /* Disable MMU stuff and caches */
  93. mrc_sctlr(val);
  94. val &= ~((0x2 << 12) | 0x7);
  95. val |= ((0x1 << 12) | (0x8 << 8) | 0x2);
  96. mcr_sctlr(val);
  97. /* CPU state is hotplug or reset */
  98. secondary_cpu_start();
  99. /* Core should not enter into WFI here */
  100. wfi();
  101. }
  102. /*
  103. * Pointer to this function is stored in iRam which is used
  104. * for jump and power down of a specific core.
  105. */
  106. static void power_down_core(void)
  107. {
  108. uint32_t tmp, core_id, core_config;
  109. /* Get the unique core id */
  110. /*
  111. * Multiprocessor Affinity Register
  112. * [11:8] Cluster ID
  113. * [1:0] CPU ID
  114. */
  115. mrc_mpafr(core_id);
  116. tmp = core_id & 0x3;
  117. core_id = (core_id >> 6) & ~3;
  118. core_id |= tmp;
  119. core_id &= 0x3f;
  120. /* Set the status of the core to low */
  121. core_config = (core_id * CPU_CONFIG_STATUS_OFFSET);
  122. core_config += EXYNOS5420_CPU_CONFIG_BASE;
  123. writel(0x0, core_config);
  124. /* Core enter WFI */
  125. wfi();
  126. }
  127. /*
  128. * Configurations for secondary cores are inapt at this stage.
  129. * Reconfigure secondary cores. Shutdown and change the status
  130. * of all cores except the primary core.
  131. */
  132. static void secondary_cores_configure(void)
  133. {
  134. /* Clear secondary boot iRAM base */
  135. writel(0x0, (CONFIG_EXYNOS_RELOCATE_CODE_BASE + 0x1C));
  136. /* set lowpower flag and address */
  137. writel(CPU_RST_FLAG_VAL, CONFIG_LOWPOWER_FLAG);
  138. writel((uint32_t)&low_power_start, CONFIG_LOWPOWER_ADDR);
  139. writel(CPU_RST_FLAG_VAL, EXYNOS5420_SPARE_BASE);
  140. /* Store jump address for power down */
  141. writel((uint32_t)&power_down_core, CONFIG_PHY_IRAM_BASE + 0x4);
  142. /* Need all core power down check */
  143. dsb();
  144. sev();
  145. }
  146. extern void relocate_wait_code(void);
  147. #endif
  148. int do_lowlevel_init(void)
  149. {
  150. uint32_t reset_status;
  151. int actions = 0;
  152. arch_cpu_init();
  153. #if !defined(CONFIG_SYS_L2CACHE_OFF) && defined(CONFIG_EXYNOS5420)
  154. /*
  155. * Init L2 cache parameters here for use by boot and resume
  156. *
  157. * These are here instead of in v7_outer_cache_enable() so that the
  158. * L2 cache settings get properly set even at resume time or if we're
  159. * running U-Boot with the cache off. The kernel still needs us to
  160. * set these for it.
  161. */
  162. configure_l2_ctlr();
  163. configure_l2_actlr();
  164. dsb();
  165. isb();
  166. relocate_wait_code();
  167. /* Reconfigure secondary cores */
  168. secondary_cores_configure();
  169. #endif
  170. reset_status = get_reset_status();
  171. switch (reset_status) {
  172. case S5P_CHECK_SLEEP:
  173. actions = DO_CLOCKS | DO_WAKEUP;
  174. break;
  175. case S5P_CHECK_DIDLE:
  176. case S5P_CHECK_LPA:
  177. actions = DO_WAKEUP;
  178. break;
  179. default:
  180. /* This is a normal boot (not a wake from sleep) */
  181. actions = DO_CLOCKS | DO_MEM_RESET | DO_POWER;
  182. }
  183. if (actions & DO_POWER)
  184. set_ps_hold_ctrl();
  185. if (actions & DO_CLOCKS) {
  186. system_clock_init();
  187. #ifdef CONFIG_DEBUG_UART
  188. #if (defined(CONFIG_SPL_BUILD) && defined(CONFIG_SPL_SERIAL_SUPPORT)) || \
  189. !defined(CONFIG_SPL_BUILD)
  190. exynos_pinmux_config(PERIPH_ID_UART3, PINMUX_FLAG_NONE);
  191. debug_uart_init();
  192. #endif
  193. #endif
  194. mem_ctrl_init(actions & DO_MEM_RESET);
  195. tzpc_init();
  196. }
  197. return actions & DO_WAKEUP;
  198. }