entry_point.S 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. /*
  2. * Copyright (c) 2006-2020, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Date Author Notes
  7. * 2020-01-15 bigmagic the first version
  8. */
  9. #include "rtconfig.h"
  10. .section ".text.entrypoint","ax"
  11. .set EL1_stack, __el1_stack
  12. .global __start
  13. // This symbol is set to 0x80000 in ld script. That is the address that raspi3's firmware
  14. // loads 'kernel8.img' file in.
  15. __start:
  16. // read cpu id, stop slave cores
  17. mrs x1, mpidr_el1 // MPIDR_EL1: Multi-Processor Affinity Register
  18. and x1, x1, #3
  19. cbz x1, .L__cpu_0 // .L prefix is the local label in ELF
  20. // cpu id > 0, stop
  21. // cpu id == 0 will also goto here after returned from entry() if possible
  22. .L__current_cpu_idle:
  23. wfe
  24. b .L__current_cpu_idle
  25. .L__cpu_0: // cpu id == 0
  26. // set stack before our code
  27. /* Define stack pointer for current exception level */
  28. // ldr x2, =EL1_stack
  29. // mov sp, x2
  30. adr x1, __start
  31. // set up EL1
  32. mrs x0, CurrentEL // CurrentEL Register. bit 2, 3. Others reserved
  33. and x0, x0, #12 // clear reserved bits
  34. // running at EL3?
  35. cmp x0, #12 // 1100b. So, EL3
  36. bne .L__not_in_el3 // 11? !EL3 -> 5:
  37. // should never be executed, just for completeness. (EL3)
  38. mov x2, #0x5b1
  39. msr scr_el3, x2 // SCR_ELn Secure Configuration Register
  40. mov x2, #0x3c9
  41. msr spsr_el3, x2 // SPSR_ELn. Saved Program Status Register. 1111001001
  42. adr x2, .L__not_in_el3
  43. msr elr_el3, x2
  44. eret // Exception Return: from EL3, continue from .L__not_in_el3
  45. // running at EL2 or EL1
  46. .L__not_in_el3:
  47. cmp x0, #4 // 0x04 0100 EL1
  48. beq .L__in_el1 // EL1 -> 5:
  49. mrs x0, hcr_el2
  50. bic x0, x0, #0xff
  51. msr hcr_el2, x0
  52. // in EL2
  53. msr sp_el1, x1 // Set sp of EL1 to _start
  54. // enable CNTP for EL1
  55. mrs x0, cnthctl_el2 // Counter-timer Hypervisor Control register
  56. orr x0, x0, #3
  57. msr cnthctl_el2, x0
  58. msr cntvoff_el2, xzr
  59. // enable AArch64 in EL1
  60. mov x0, #(1 << 31) // AArch64
  61. orr x0, x0, #(1 << 1) // SWIO hardwired on Pi3
  62. msr hcr_el2, x0
  63. mrs x0, hcr_el2
  64. // change execution level to EL1
  65. mov x2, #0x3c4
  66. msr spsr_el2, x2 // 1111000100
  67. adr x2, .L__in_el1
  68. msr elr_el2, x2
  69. eret // exception return. from EL2. continue from .L__in_el1
  70. .L__in_el1:
  71. ldr x9, =PV_OFFSET
  72. mov sp, x1 // in EL1. Set sp to _start
  73. // Set CPACR_EL1 (Architecture Feature Access Control Register) to avoid trap from SIMD or float point instruction
  74. mov x1, #0x00300000 // Don't trap any SIMD/FP instructions in both EL0 and EL1
  75. msr cpacr_el1, x1
  76. // clear bss
  77. ldr x1, =__bss_start
  78. add x1, x1, x9
  79. ldr w2, =__bss_size
  80. .L__clean_bss_loop:
  81. cbz w2, .L__jump_to_entry
  82. str xzr, [x1], #8
  83. sub w2, w2, #1
  84. cbnz w2, .L__clean_bss_loop
  85. // jump to C code, should not return
  86. .L__jump_to_entry:
  87. bl get_free_page
  88. mov x21, x0
  89. bl get_free_page
  90. mov x20, x0
  91. mov x1, x21
  92. bl mmu_tcr_init
  93. mov x0, x20
  94. mov x1, x21
  95. msr ttbr0_el1, x0
  96. msr ttbr1_el1, x1
  97. dsb sy
  98. ldr x2, =0x40000000 //1G
  99. mov x3, x9
  100. bl rt_hw_mmu_setup_early
  101. ldr x30, =after_mmu_enable
  102. mrs x1, sctlr_el1
  103. bic x1, x1, #(3 << 3) /* dis SA, SA0 */
  104. bic x1, x1, #(1 << 1) /* dis A */
  105. orr x1, x1, #(1 << 12) /* I */
  106. orr x1, x1, #(1 << 2) /* C */
  107. orr x1, x1, #(1 << 0) /* M */
  108. msr sctlr_el1, x1
  109. dsb sy
  110. isb sy
  111. ic ialluis
  112. dsb sy
  113. isb sy
  114. tlbi vmalle1
  115. dsb sy
  116. isb sy
  117. ret
  118. after_mmu_enable:
  119. // disable ttbr0
  120. mrs x0, tcr_el1
  121. orr x0, x0, #(1 << 7)
  122. msr tcr_el1, x0
  123. msr ttbr0_el1, xzr
  124. dsb sy
  125. mov x0, #1
  126. msr spsel, x0
  127. adr x1, __start
  128. mov sp, x1 // sp_el1 set to _start
  129. b rtthread_startup