summaryrefslogtreecommitdiff
path: root/ldso/ldso/cris/resolve.S
blob: 9ca27b5efa9d89d22a15e3abe2238e2def98cb2a (plain)
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
/*
 * This function is _not_ called directly.  It is jumped to from PLT when
 * attempting to use a symbol that has not yet been resolved.  The first time a
 * jump symbol (such as a function call inside a shared library) is used
 * (before it gets resolved) it will jump here.  When we get called the stack
 * contains reloc_offset and tpnt is in MOF.
 *
 * We save all the registers, setup R10 and R11 with the right arguments then
 * call _dl_linux_resolver(tpnt, reloc_offset). _dl_linux_resolver() figures
 * out where the jump symbol is _really_ supposed to have jumped to and returns
 * that to us.  Once we have that, we overwrite tpnt with this fixed up
 * address. We then clean up after ourselves, put all the registers back how we
 * found them, then we jump to where the fixed up address, which is where the
 * jump symbol that got us here really wanted to jump to in the first place.
 */

.globl _dl_linux_resolve
.type _dl_linux_resolve,@function

#ifdef __arch_v32

_dl_linux_resolve:
	subq	4,$sp
	move.d	$r0,[$sp]
	subq	4,$sp
	move.d	$r13,[$sp]
	subq	4,$sp
	move.d	$r12,[$sp]
	subq	4,$sp
	move.d	$r11,[$sp]
	subq	4,$sp
	addoq	5*4,$sp,$acr
	move.d	$r10,[$sp]
	subq	4,$sp
	move	$mof,$r10
	move.d	$r9,[$sp]
	subq	4,$sp
	move.d	[$acr],$r11
	move	$srp,[$sp]
	lapc	_GLOBAL_OFFSET_TABLE_,$r0
	move.d	_dl_linux_resolver:PLTG,$r9
	add.d	$r0,$r9
	jsr	$r9
	nop
	move.d	$r10,$acr
	move	[$sp+],$srp
	move.d	[$sp+],$r9
	move.d	[$sp+],$r10
	move.d	[$sp+],$r11
	move.d	[$sp+],$r12
	move.d	[$sp+],$r13
	move.d	[$sp+],$r0
	jump	$acr
	addq	4,$sp

#else

_dl_linux_resolve:
	push	$r13
	push	$r12
	push	$r11
	push	$r10
	push	$r9
	push	$r0
	push	$srp
	move.d	[$sp+7*4],$r11
	move	$mof,$r10
#ifdef __PIC__
	move.d	$pc,$r0
	sub.d	.:GOTOFF,$r0
	move.d	_dl_linux_resolver:PLTG,$r9
	add.d	$r0,$r9
	jsr	$r9
#else
	jsr	_dl_linux_resolver
#endif
	move.d	$r10,[$sp+7*4]
	pop	$srp
	pop	$r0
	pop	$r9
	pop	$r10
	pop	$r11
	pop	$r12
	pop	$r13
	jump	[$sp+]

#endif /* __arch_v32 */

	.size _dl_linux_resolve, . - _dl_linux_resolve