entry_point.S 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. .section ".text.entrypoint"
  2. .set EL1_stack, __el1_stack
  3. .global _start
  4. // This symbol is set to 0x80000 in ld script. That is the address that raspi3's firmware
  5. // loads 'kernel8.img' file in.
  6. _start:
  7. // read cpu id, stop slave cores
  8. mrs x1, mpidr_el1 // MPIDR_EL1: Multi-Processor Affinity Register
  9. and x1, x1, #3
  10. cbz x1, .L__cpu_0 // .L prefix is the local label in ELF
  11. // cpu id > 0, stop
  12. // cpu id == 0 will also goto here after returned from entry() if possible
  13. .L__current_cpu_idle:
  14. wfe
  15. b .L__current_cpu_idle
  16. .L__cpu_0: // cpu id == 0
  17. // set stack before our code
  18. /* Define stack pointer for current exception level */
  19. // ldr x2, =EL1_stack
  20. // mov sp, x2
  21. ldr x1, =_start
  22. // set up EL1
  23. mrs x0, CurrentEL // CurrentEL Register. bit 2, 3. Others reserved
  24. and x0, x0, #12 // clear reserved bits
  25. // running at EL3?
  26. cmp x0, #12 // 1100b. So, EL3
  27. bne .L__not_in_el3 // 11? !EL3 -> 5:
  28. // should never be executed, just for completeness. (EL3)
  29. mov x2, #0x5b1
  30. msr scr_el3, x2 // SCR_ELn Secure Configuration Register
  31. mov x2, #0x3c9
  32. msr spsr_el3, x2 // SPSR_ELn. Saved Program Status Register. 1111001001
  33. adr x2, .L__not_in_el3
  34. msr elr_el3, x2
  35. eret // Exception Return: from EL3, continue from .L__not_in_el3
  36. // running at EL2 or EL1
  37. .L__not_in_el3:
  38. cmp x0, #4 // 0x04 0100 EL1
  39. beq .L__in_el1 // EL1 -> 5:
  40. // in EL2
  41. msr sp_el1, x1 // Set sp of EL1 to _start
  42. // enable CNTP for EL1
  43. mrs x0, cnthctl_el2 // Counter-timer Hypervisor Control register
  44. orr x0, x0, #3
  45. msr cnthctl_el2, x0
  46. msr cntvoff_el2, xzr
  47. // enable AArch64 in EL1
  48. mov x0, #(1 << 31) // AArch64
  49. orr x0, x0, #(1 << 1) // SWIO hardwired on Pi3
  50. msr hcr_el2, x0
  51. mrs x0, hcr_el2
  52. // change execution level to EL1
  53. mov x2, #0x3c4
  54. msr spsr_el2, x2 // 1111000100
  55. adr x2, .L__in_el1
  56. msr elr_el2, x2
  57. eret // exception return. from EL2. continue from .L__in_el1
  58. .L__in_el1:
  59. mov sp, x1 // in EL1. Set sp to _start
  60. // Set CPACR_EL1 (Architecture Feature Access Control Register) to avoid trap from SIMD or float point instruction
  61. mov x1, #0x00300000 // Don't trap any SIMD/FP instructions in both EL0 and EL1
  62. msr cpacr_el1, x1
  63. mrs x1, sctlr_el1
  64. orr x1, x1, #(1 << 12)
  65. bic x1, x1, #(3 << 3)
  66. bic x1, x1, #(1 << 1)
  67. msr sctlr_el1, x1
  68. // clear bss
  69. ldr x1, =__bss_start
  70. ldr w2, =__bss_size
  71. .L__clean_bss_loop:
  72. cbz w2, .L__jump_to_entry
  73. str xzr, [x1], #8
  74. sub w2, w2, #1
  75. cbnz w2, .L__clean_bss_loop
  76. // jump to C code, should not return
  77. .L__jump_to_entry:
  78. bl entry
  79. // for failsafe, halt this core too
  80. b .L__current_cpu_idle