Commit e31aa453 authored by Paul Mackerras's avatar Paul Mackerras

powerpc: Use LOAD_REG_IMMEDIATE only for constants on 64-bit

Using LOAD_REG_IMMEDIATE to get the address of kernel symbols
generates 5 instructions where LOAD_REG_ADDR can do it in one,
and will generate R_PPC64_ADDR16_* relocations in the output when
we get to making the kernel as a position-independent executable,
which we'd rather not have to handle.  This changes various bits
of assembly code to use LOAD_REG_ADDR when we need to get the
address of a symbol, or to use suitable position-independent code
for cases where we can't access the TOC for various reasons, or
if we're not running at the address we were linked at.

It also cleans up a few minor things; there's no reason to save and
restore SRR0/1 around RTAS calls, __mmu_off can get the return
address from LR more conveniently than the caller can supply it in
R4 (and we already assume elsewhere that EA == RA if the MMU is on
in early boot), and enable_64b_mode was using 5 instructions where
2 would do.
Signed-off-by: default avatarPaul Mackerras <paulus@samba.org>
parent 1f6a93e4
...@@ -268,7 +268,7 @@ GLUE(.,name): ...@@ -268,7 +268,7 @@ GLUE(.,name):
* Loads the value of the constant expression 'expr' into register 'rn' * Loads the value of the constant expression 'expr' into register 'rn'
* using immediate instructions only. Use this when it's important not * using immediate instructions only. Use this when it's important not
* to reference other data (i.e. on ppc64 when the TOC pointer is not * to reference other data (i.e. on ppc64 when the TOC pointer is not
* valid). * valid) and when 'expr' is a constant or absolute address.
* *
* LOAD_REG_ADDR(rn, name) * LOAD_REG_ADDR(rn, name)
* Loads the address of label 'name' into register 'rn'. Use this when * Loads the address of label 'name' into register 'rn'. Use this when
......
...@@ -110,7 +110,7 @@ load_hids: ...@@ -110,7 +110,7 @@ load_hids:
isync isync
/* Save away cpu state */ /* Save away cpu state */
LOAD_REG_IMMEDIATE(r5,cpu_state_storage) LOAD_REG_ADDR(r5,cpu_state_storage)
/* Save HID0,1,4 and 5 */ /* Save HID0,1,4 and 5 */
mfspr r3,SPRN_HID0 mfspr r3,SPRN_HID0
...@@ -134,7 +134,7 @@ _GLOBAL(__restore_cpu_ppc970) ...@@ -134,7 +134,7 @@ _GLOBAL(__restore_cpu_ppc970)
rldicl. r0,r0,4,63 rldicl. r0,r0,4,63
beqlr beqlr
LOAD_REG_IMMEDIATE(r5,cpu_state_storage) LOAD_REG_ADDR(r5,cpu_state_storage)
/* Before accessing memory, we make sure rm_ci is clear */ /* Before accessing memory, we make sure rm_ci is clear */
li r0,0 li r0,0
mfspr r3,SPRN_HID4 mfspr r3,SPRN_HID4
......
...@@ -690,10 +690,6 @@ _GLOBAL(enter_rtas) ...@@ -690,10 +690,6 @@ _GLOBAL(enter_rtas)
std r7,_DAR(r1) std r7,_DAR(r1)
mfdsisr r8 mfdsisr r8
std r8,_DSISR(r1) std r8,_DSISR(r1)
mfsrr0 r9
std r9,_SRR0(r1)
mfsrr1 r10
std r10,_SRR1(r1)
/* Temporary workaround to clear CR until RTAS can be modified to /* Temporary workaround to clear CR until RTAS can be modified to
* ignore all bits. * ignore all bits.
...@@ -754,6 +750,10 @@ _STATIC(rtas_return_loc) ...@@ -754,6 +750,10 @@ _STATIC(rtas_return_loc)
mfspr r4,SPRN_SPRG3 /* Get PACA */ mfspr r4,SPRN_SPRG3 /* Get PACA */
clrldi r4,r4,2 /* convert to realmode address */ clrldi r4,r4,2 /* convert to realmode address */
bcl 20,31,$+4
0: mflr r3
ld r3,(1f-0b)(r3) /* get &.rtas_restore_regs */
mfmsr r6 mfmsr r6
li r0,MSR_RI li r0,MSR_RI
andc r6,r6,r0 andc r6,r6,r0
...@@ -761,7 +761,6 @@ _STATIC(rtas_return_loc) ...@@ -761,7 +761,6 @@ _STATIC(rtas_return_loc)
mtmsrd r6 mtmsrd r6
ld r1,PACAR1(r4) /* Restore our SP */ ld r1,PACAR1(r4) /* Restore our SP */
LOAD_REG_IMMEDIATE(r3,.rtas_restore_regs)
ld r4,PACASAVEDMSR(r4) /* Restore our MSR */ ld r4,PACASAVEDMSR(r4) /* Restore our MSR */
mtspr SPRN_SRR0,r3 mtspr SPRN_SRR0,r3
...@@ -769,6 +768,9 @@ _STATIC(rtas_return_loc) ...@@ -769,6 +768,9 @@ _STATIC(rtas_return_loc)
rfid rfid
b . /* prevent speculative execution */ b . /* prevent speculative execution */
.align 3
1: .llong .rtas_restore_regs
_STATIC(rtas_restore_regs) _STATIC(rtas_restore_regs)
/* relocation is on at this point */ /* relocation is on at this point */
REST_GPR(2, r1) /* Restore the TOC */ REST_GPR(2, r1) /* Restore the TOC */
...@@ -788,10 +790,6 @@ _STATIC(rtas_restore_regs) ...@@ -788,10 +790,6 @@ _STATIC(rtas_restore_regs)
mtdar r7 mtdar r7
ld r8,_DSISR(r1) ld r8,_DSISR(r1)
mtdsisr r8 mtdsisr r8
ld r9,_SRR0(r1)
mtsrr0 r9
ld r10,_SRR1(r1)
mtsrr1 r10
addi r1,r1,RTAS_FRAME_SIZE /* Unstack our frame */ addi r1,r1,RTAS_FRAME_SIZE /* Unstack our frame */
ld r0,16(r1) /* get return address */ ld r0,16(r1) /* get return address */
......
This diff is collapsed.
...@@ -31,11 +31,14 @@ _GLOBAL(reloc_offset) ...@@ -31,11 +31,14 @@ _GLOBAL(reloc_offset)
mflr r0 mflr r0
bl 1f bl 1f
1: mflr r3 1: mflr r3
LOAD_REG_IMMEDIATE(r4,1b) PPC_LL r4,(2f-1b)(r3)
subf r3,r4,r3 subf r3,r4,r3
mtlr r0 mtlr r0
blr blr
.align 3
2: PPC_LONG 1b
/* /*
* add_reloc_offset(x) returns x + reloc_offset(). * add_reloc_offset(x) returns x + reloc_offset().
*/ */
...@@ -43,12 +46,15 @@ _GLOBAL(add_reloc_offset) ...@@ -43,12 +46,15 @@ _GLOBAL(add_reloc_offset)
mflr r0 mflr r0
bl 1f bl 1f
1: mflr r5 1: mflr r5
LOAD_REG_IMMEDIATE(r4,1b) PPC_LL r4,(2f-1b)(r5)
subf r5,r4,r5 subf r5,r4,r5
add r3,r3,r5 add r3,r3,r5
mtlr r0 mtlr r0
blr blr
.align 3
2: PPC_LONG 1b
_GLOBAL(kernel_execve) _GLOBAL(kernel_execve)
li r0,__NR_execve li r0,__NR_execve
sc sc
......
...@@ -38,12 +38,13 @@ ...@@ -38,12 +38,13 @@
.globl system_reset_iSeries .globl system_reset_iSeries
system_reset_iSeries: system_reset_iSeries:
bl .relative_toc
mfspr r13,SPRN_SPRG3 /* Get alpaca address */ mfspr r13,SPRN_SPRG3 /* Get alpaca address */
LOAD_REG_IMMEDIATE(r23, alpaca) LOAD_REG_ADDR(r23, alpaca)
li r0,ALPACA_SIZE li r0,ALPACA_SIZE
sub r23,r13,r23 sub r23,r13,r23
divdu r23,r23,r0 /* r23 has cpu number */ divdu r23,r23,r0 /* r23 has cpu number */
LOAD_REG_IMMEDIATE(r13, paca) LOAD_REG_ADDR(r13, paca)
mulli r0,r23,PACA_SIZE mulli r0,r23,PACA_SIZE
add r13,r13,r0 add r13,r13,r0
mtspr SPRN_SPRG3,r13 /* Save it away for the future */ mtspr SPRN_SPRG3,r13 /* Save it away for the future */
...@@ -60,14 +61,14 @@ system_reset_iSeries: ...@@ -60,14 +61,14 @@ system_reset_iSeries:
mtspr SPRN_CTRLT,r4 mtspr SPRN_CTRLT,r4
/* Spin on __secondary_hold_spinloop until it is updated by the boot cpu. */ /* Spin on __secondary_hold_spinloop until it is updated by the boot cpu. */
/* In the UP case we'll yeild() later, and we will not access the paca anyway */ /* In the UP case we'll yield() later, and we will not access the paca anyway */
#ifdef CONFIG_SMP #ifdef CONFIG_SMP
1: 1:
HMT_LOW HMT_LOW
LOAD_REG_IMMEDIATE(r23, __secondary_hold_spinloop) LOAD_REG_ADDR(r23, __secondary_hold_spinloop)
ld r23,0(r23) ld r23,0(r23)
sync sync
LOAD_REG_IMMEDIATE(r3,current_set) LOAD_REG_ADDR(r3,current_set)
sldi r28,r24,3 /* get current_set[cpu#] */ sldi r28,r24,3 /* get current_set[cpu#] */
ldx r3,r3,r28 ldx r3,r3,r28
addi r1,r3,THREAD_SIZE addi r1,r3,THREAD_SIZE
...@@ -90,7 +91,7 @@ system_reset_iSeries: ...@@ -90,7 +91,7 @@ system_reset_iSeries:
lbz r23,PACAPROCSTART(r13) /* Test if this processor lbz r23,PACAPROCSTART(r13) /* Test if this processor
* should start */ * should start */
sync sync
LOAD_REG_IMMEDIATE(r3,current_set) LOAD_REG_ADDR(r3,current_set)
sldi r28,r24,3 /* get current_set[cpu#] */ sldi r28,r24,3 /* get current_set[cpu#] */
ldx r3,r3,r28 ldx r3,r3,r28
addi r1,r3,THREAD_SIZE addi r1,r3,THREAD_SIZE
...@@ -255,8 +256,8 @@ hardware_interrupt_iSeries_masked: ...@@ -255,8 +256,8 @@ hardware_interrupt_iSeries_masked:
_INIT_STATIC(__start_initialization_iSeries) _INIT_STATIC(__start_initialization_iSeries)
/* Clear out the BSS */ /* Clear out the BSS */
LOAD_REG_IMMEDIATE(r11,__bss_stop) LOAD_REG_ADDR(r11,__bss_stop)
LOAD_REG_IMMEDIATE(r8,__bss_start) LOAD_REG_ADDR(r8,__bss_start)
sub r11,r11,r8 /* bss size */ sub r11,r11,r8 /* bss size */
addi r11,r11,7 /* round up to an even double word */ addi r11,r11,7 /* round up to an even double word */
rldicl. r11,r11,61,3 /* shift right by 3 */ rldicl. r11,r11,61,3 /* shift right by 3 */
...@@ -267,15 +268,11 @@ _INIT_STATIC(__start_initialization_iSeries) ...@@ -267,15 +268,11 @@ _INIT_STATIC(__start_initialization_iSeries)
3: stdu r0,8(r8) 3: stdu r0,8(r8)
bdnz 3b bdnz 3b
4: 4:
LOAD_REG_IMMEDIATE(r1,init_thread_union) LOAD_REG_ADDR(r1,init_thread_union)
addi r1,r1,THREAD_SIZE addi r1,r1,THREAD_SIZE
li r0,0 li r0,0
stdu r0,-STACK_FRAME_OVERHEAD(r1) stdu r0,-STACK_FRAME_OVERHEAD(r1)
LOAD_REG_IMMEDIATE(r2,__toc_start)
addi r2,r2,0x4000
addi r2,r2,0x4000
bl .iSeries_early_setup bl .iSeries_early_setup
bl .early_setup bl .early_setup
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment