crt1.S 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. /* When we enter this piece of code, the program stack looks like this:
  2. argc argument counter (integer)
  3. argv[0] program name (pointer)
  4. argv[1...N] program args (pointers)
  5. argv[argc-1] end of args (integer)
  6. NULL
  7. env[0...N] environment variables (pointers)
  8. NULL
  9. For uClinux it looks like this:
  10. argc argument counter (integer)
  11. argv char *argv[]
  12. envp char *envp[]
  13. argv[0] program name (pointer)
  14. argv[1...N] program args (pointers)
  15. argv[argc-1] end of args (integer)
  16. NULL
  17. env[0...N] environment variables (pointers)
  18. NULL
  19. When we are done here, we want
  20. a1=argc
  21. a2=argv[0]
  22. a3=argv[argc+1]
  23. ARM register quick reference:
  24. Name Number ARM Procedure Calling Standard Role
  25. a1 r0 argument 1 / integer result / scratch register / argc
  26. a2 r1 argument 2 / scratch register / argv
  27. a3 r2 argument 3 / scratch register / envp
  28. a4 r3 argument 4 / scratch register
  29. v1 r4 register variable
  30. v2 r5 register variable
  31. v3 r6 register variable
  32. v4 r7 register variable
  33. v5 r8 register variable
  34. sb/v6 r9 static base / register variable
  35. sl/v7 r10 stack limit / stack chunk handle / reg. variable
  36. fp r11 frame pointer
  37. ip r12 scratch register / new-sb in inter-link-unit calls
  38. sp r13 lower end of current stack frame
  39. lr r14 link address / scratch register
  40. pc r15 program counter
  41. */
  42. #include <features.h>
  43. .text
  44. .global _start
  45. .type _start,%function
  46. .type _init,%function
  47. .type _fini,%function
  48. .type main,%function
  49. .type __uClibc_main,%function
  50. .text
  51. _start:
  52. /* Save a copy of rtld_fini before r0 gets nuked */
  53. mov r5, r0
  54. /* clear the frame pointer */
  55. mov fp, #0
  56. /* Load register r0 with main */
  57. #ifdef __PIC__
  58. adr r8, .L_main
  59. ldr r0, .L_main
  60. add r0, r0, r8
  61. ldr r4, .L_init + 4
  62. add r4, r4, r8
  63. #else
  64. ldr r0, =main
  65. #endif
  66. #ifdef __ARCH_HAS_MMU__
  67. /* Load register r1 from the stack to its final resting place */
  68. ldr r1, [sp], #4
  69. /* Copy argv pointer into r2 -- which its final resting place */
  70. mov r2, sp
  71. #else
  72. /*
  73. * uClinux stacks look a little different from normal
  74. * MMU-full Linux stacks (for no good reason)
  75. */
  76. /* pull argc and argv off the stack */
  77. ldr r1,[sp, #0]
  78. ldr r2,[sp, #4]
  79. #endif
  80. /* Store _init and _fini to r3 and r4 */
  81. #ifdef __PIC__
  82. adr r8, .L_init
  83. ldr r3, .L_init
  84. add r3, r3, r8
  85. ldr r4, .L_init + 4
  86. add r4, r4, r8
  87. #else
  88. ldr r3, =_init
  89. ldr r4, =_fini
  90. #endif
  91. /* Store _fini(r4), rtld_fini(r5), and stack_end(r2) on the stack */
  92. str r2, [sp, #-4]!
  93. str r5, [sp, #-4]!
  94. str r4, [sp, #-4]!
  95. /* We need to call __uClibc_main which should not return.
  96. __uClibc_main (int (*main) (int, char **, char **), int argc,
  97. char **argv, void (*init) (void), void (*fini) (void),
  98. void (*rtld_fini) (void), void *stack_end)
  99. */
  100. bl __uClibc_main
  101. /* Crash if somehow `exit' returns anyways. */
  102. bl abort
  103. #ifdef __PIC__
  104. .L_init:
  105. .word _init
  106. .word _fini
  107. .L_main:
  108. .word main
  109. #endif
  110. /* We need this stuff to make gdb behave itself, otherwise
  111. gdb will choke with SIGILL when trying to debug apps.
  112. */
  113. .section ".note.ABI-tag", "a"
  114. .align 4
  115. .long 1f - 0f
  116. .long 3f - 2f
  117. .long 1
  118. 0: .asciz "GNU"
  119. 1: .align 4
  120. 2: .long 0
  121. .long 2,0,0
  122. 3: .align 4
  123. /* Define a symbol for the first piece of initialized data. */
  124. .data
  125. .globl __data_start
  126. __data_start:
  127. .long 0
  128. .weak data_start
  129. data_start = __data_start