1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
|
/*
* CDDL HEADER START
*
* The contents of this file are subject to the terms of the
* Common Development and Distribution License (the "License").
* You may not use this file except in compliance with the License.
*
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
* or http://www.opensolaris.org/os/licensing.
* See the License for the specific language governing permissions
* and limitations under the License.
*
* When distributing Covered Code, include this CDDL HEADER in each
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
* If applicable, add the following below this CDDL HEADER, with the
* fields enclosed by brackets "[]" replaced with your own identifying
* information: Portions Copyright [yyyy] [name of copyright owner]
*
* CDDL HEADER END
*/
/*
* Copyright (c) 1988 AT&T
* All Rights Reserved
*
* Copyright 2007 Sun Microsystems, Inc. All rights reserved.
* Use is subject to license terms.
*/
#pragma ident "%Z%%M% %I% %E% SMI"
/*
* Bootstrap routine for ld.so. Control arrives here directly from
* exec() upon invocation of a dynamically linked program specifying ld.so
* as its interpreter.
*
* On entry, the stack appears as:
*
* !_______________________! high addresses
* ! 0 word !
* !_______________________!
* ! !
* ! Information !
* ! Block !
* ! (size varies) !
* !_______________________!
* ! Auxiliary !
* ! vector !
* ! 2 word entries !
* ! !
* !_______________________!
* ! 0 word !
* !_______________________!
* ! Environment !
* ! pointers !
* ! ... !
* ! (one word each) !
* !_______________________!
* ! 0 word !
* !_______________________!
* ! Argument ! low addresses
* ! pointers !
* ! Argc words !
* !_______________________!
* ! !
* ! Argc !
* !_______________________! <- %sp + STACK_BIAS + WINDOWSIZE
* ! !
* ! Window save area !
* !_______________________! <- %sp + STACK_BIAS
*/
#if defined(lint)
extern unsigned long _setup();
extern void atexit_fini();
void
main()
{
(void) _setup();
atexit_fini();
}
#else
#include <sys/asm_linkage.h>
#include <sys/param.h>
#include <link.h>
.file "boot.s"
.seg ".text"
.global _rt_boot, _setup, atexit_fini
.type _rt_boot, #function
.align 4
! Entry vector
! +0: normal start
! +4: normal start
! +8: alias start (frame exists) XX64 what's this for?
_rt_boot:
nop
ba,a _elf_start
ba,a _alias_start
! Start up routines
_elf_start:
! Create a stack frame, perform PIC set up. We have
! to determine a bunch of things from our "environment" and
! construct an Elf64_Boot attribute value vector.
save %sp, -SA(MINFRAME + (EB_MAX * 16)), %sp
_alias_start:
1: ! PIC prologue
call 2f
sethi %hh(_GLOBAL_OFFSET_TABLE_ + (. - 1b)), %g1
2: or %g1, %hm(_GLOBAL_OFFSET_TABLE_ + (. - 1b)), %g1
sllx %g1, 32, %g5
sethi %lm(_GLOBAL_OFFSET_TABLE_ + (. - 1b)), %l7
or %l7, %lo(_GLOBAL_OFFSET_TABLE_ + (. - 1b)), %l7
or %g5, %l7, %l7
add %l7, %o7, %l7 ! finish PIC prologue
! %fp points to the root of our ELF bootstrap vector, use it to construct
! the vector and send it to _setup.
!
! The resulting Elf64_Boot vector looks like this:
!
! Offset Contents
! +0x0 EB_ARGV
! +0x8 argv[]
! +0x10 EB_ENVP
! +0x18 envp[]
! +0x20 EB_AUXV
! +0x28 auxv[]
! +0x30 EB_NULL
add %sp, STACK_BIAS + SA(MINFRAME), %o0
! &eb[0] == %sp + frame size
mov EB_ARGV, %l0 ! code for this entry
stx %l0, [%o0] ! store it
add %fp, WINDOWSIZE + 8 + STACK_BIAS, %l0
! argument vector
stx %l0, [%o0 + 0x8] ! store that
ldx [%fp + WINDOWSIZE + STACK_BIAS], %l1
! get argument count (argc)
inc %l1 ! account for last element of 0
sllx %l1, 3, %l1 ! multiply by 8
add %l0, %l1, %l0 ! and get address of first env ptr
stx %l0, [%o0 + 0x18] ! store it in the vector
mov EB_ENVP, %l1 ! code for environment base
stx %l1, [%o0 + 0x10] ! store it
mov EB_AUXV, %l1 ! get code for auxiliary vector
stx %l1, [%o0 + 0x20] ! store it
3: ldx [%l0], %l1 ! get an entry
brnz,pt %l1, 3b ! if not at end, go back and look again
add %l0, 8, %l0 ! incrementing pointer in delay
stx %l0, [%o0 + 0x28] ! store aux vector pointer
mov EB_NULL, %l0 ! set up for the last pointer
stx %l0, [%o0 + 0x30] ! and store it
mov %g0, %g2 ! clear globals
mov %g0, %g3
! Call _setup. Two arguments, the ELF bootstrap vector and our (unrelocated)
! _DYNAMIC address. The _DYNAMIC address is located in entry 0 of the GOT
call _setup ! call it
ldx [%l7], %o1
! On return, give callee the exit function in %g1, and jump to the
! target program, clearing out the reserved globals as we go.
ldx [%l7 + atexit_fini], %g1! get function address
restore %o0, %g0, %l1 ! release frame
jmpl %l1, %g0 ! call main program
mov %g0, %g4 ! clear one last global in delay
.size _rt_boot, . - _rt_boot
#endif
|