sparc: leon3: Updates for generic board initialization

Reworked the LEON3 start.S code to call board_init_f function at startup.
Also implemented the relocate_code function in assembly to relocate the
monitor and setup the stack pointer before calling relocated board_init_r.

Add the CONFIG_SYS_GENERIC_BOARD variable to all the LEON3 boards.

Signed-off-by: Francois Retief <fgretief@spaceteq.co.za>
This commit is contained in:
Francois Retief 2015-10-29 00:02:48 +02:00
parent cb31eaa4b3
commit a62bba15b5
6 changed files with 151 additions and 116 deletions

View File

@ -47,17 +47,6 @@ void cpu_init_f(void)
#endif #endif
} }
/* Routine called from start.S,
*
* Run from FLASH/PROM:
* - memory controller has already been setup up, stack can be used
* - global variables available for read/writing
* - constants avaiable
*/
void cpu_init_f2(void)
{
}
/* If cache snooping is available in hardware the result will be set /* If cache snooping is available in hardware the result will be set
* to 0x800000, otherwise 0. * to 0x800000, otherwise 0.
*/ */

View File

@ -257,11 +257,18 @@ wiminit:
set WIM_INIT, %g3 set WIM_INIT, %g3
mov %g3, %wim mov %g3, %wim
stackp: stackinit:
set CONFIG_SYS_INIT_SP_OFFSET, %fp set CONFIG_SYS_INIT_SP_OFFSET, %fp
andn %fp, 0x0f, %fp andn %fp, 0x0f, %fp
sub %fp, 64, %sp sub %fp, 64, %sp
tbrinit:
set CONFIG_SYS_TEXT_BASE, %g2
wr %g0, %g2, %tbr
nop
nop
nop
/* Obtain the address of _GLOBAL_OFFSET_TABLE_ */ /* Obtain the address of _GLOBAL_OFFSET_TABLE_ */
SPARC_PIC_THUNK_CALL(l7) SPARC_PIC_THUNK_CALL(l7)
@ -298,25 +305,50 @@ cpu_init_unreloc:
call cpu_init_f call cpu_init_f
nop nop
/* un relocated start address of monitor */ board_init_unreloc:
#define TEXT_START _text call board_init_f
clr %o0 ! boot_flags
/* un relocated end address of monitor */ dead_unreloc:
#define DATA_END __init_end mov 1, %g1 ! For GRMON2 to exit normally.
ta 0 ! If board_init_f call returns.. (unlikely)
nop
nop
ba dead_unreloc ! infinte loop
nop
!-------------------------------------------------------------------------------
/* void relocate_code (addr_sp, gd, addr_moni)
*
* This "function" does not return, instead it continues in RAM after
* relocating the monitor code.
*
* %o0 = Relocated stack pointer
* %o1 = Relocated global data pointer
* %o2 = Relocated text pointer
*
* %l7 = _GLOBAL_OFFSET_TABLE_ address
*/
.globl relocate_code
.type relocate_code, #function
.align 4
relocate_code:
!SPARC_PIC_THUNK_CALL(l7)
reloc: reloc:
SPARC_LOAD_ADDRESS(TEXT_START, l7, g2) SPARC_LOAD_ADDRESS(_text, l7, g2) ! start address of monitor
SPARC_LOAD_ADDRESS(DATA_END, l7, g3) SPARC_LOAD_ADDRESS(__init_end, l7, g3) ! end address of monitor
set CONFIG_SYS_RELOC_MONITOR_BASE,%g4 mov %o2, %g4 ! relocation address
reloc_loop: sub %g4, %g2, %g6 ! relocation offset
ldd [%g2],%l0 /* copy .text & .data to relocated address */
ldd [%g2+8],%l2 10: ldd [%g2], %l0
std %l0,[%g4] ldd [%g2+8], %l2
std %l2,[%g4+8] std %l0, [%g4]
inc 16,%g2 std %l2, [%g4+8]
subcc %g3,%g2,%g0 inc 16, %g2 ! src += 16
bne reloc_loop cmp %g2, %g3
inc 16,%g4 bcs 10b ! while (src < end)
inc 16, %g4 ! dst += 16
clr %l0 clr %l0
clr %l1 clr %l1
@ -331,49 +363,42 @@ reloc_loop:
* *
*/ */
/* clear the relocated .bss area */
clr_bss: clr_bss:
/* clear bss area (the relocated) */
SPARC_LOAD_ADDRESS(__bss_start, l7, g2) SPARC_LOAD_ADDRESS(__bss_start, l7, g2)
SPARC_LOAD_ADDRESS(__bss_end, l7, g3) SPARC_LOAD_ADDRESS(__bss_end, l7, g3)
sub %g3,%g2,%g3 sub %g3,%g2,%g3 ! length of .bss area
add %g3,%g4,%g3 add %g3,%g4,%g3
/* clearing 16byte a time ==> linker script need to align to 16 byte offset */
clr %g1 /* std %g0 uses g0 and g1 */ clr %g1 /* std %g0 uses g0 and g1 */
/* clearing 16byte a time ==> linker script need to align to 16 byte offset */ 20:
clr_bss_16: std %g0, [%g4]
std %g0,[%g4] std %g0, [%g4+8]
std %g0,[%g4+8] inc 16, %g4 ! ptr += 16
inc 16,%g4 cmp %g4, %g3
cmp %g3,%g4 bcs 20b ! while (ptr < end)
bne clr_bss_16
nop nop
/* add offsets to GOT table */ /* add offsets to GOT table */
fixup_got: fixup_got:
SPARC_LOAD_ADDRESS(__got_start, l7, g4) SPARC_LOAD_ADDRESS(__got_start, l7, g4)
add %g4, %g6, %g4
SPARC_LOAD_ADDRESS(__got_end, l7, g3) SPARC_LOAD_ADDRESS(__got_end, l7, g3)
/* add %g3, %g6, %g3
* new got offset = (old GOT-PTR (read with ld) - 30: ld [%g4], %l0
* CONFIG_SYS_RELOC_MONITOR_BASE(from define) ) + #ifdef CONFIG_RELOC_GOT_SKIP_NULL
* Destination Address (from define) cmp %l0, 0
*/ be 32f
set CONFIG_SYS_RELOC_MONITOR_BASE,%g2 #endif
SPARC_LOAD_ADDRESS(TEXT_START, l7, g1) add %l0, %g6, %l0 ! relocate GOT pointer
add %g4,%g2,%g4 st %l0, [%g4]
sub %g4,%g1,%g4 32: inc 4, %g4 ! ptr += 4
add %g3,%g2,%g3 cmp %g4, %g3
sub %g3,%g1,%g3 bcs 30b ! while (ptr < end)
sub %g2,%g1,%g2 ! prepare register with (new base address) -
! (old base address)
got_loop:
ld [%g4],%l0 ! load old GOT-PTR
add %l0,%g2,%l0 ! increase with (new base address) -
! (old base)
st %l0,[%g4]
inc 4,%g4
cmp %g3,%g4
bne got_loop
nop nop
#if 0 /* FIXME: Relocated PROM address should be calculated! */
prom_relocate: prom_relocate:
SPARC_LOAD_ADDRESS(__prom_start, l7, g2) SPARC_LOAD_ADDRESS(__prom_start, l7, g2)
SPARC_LOAD_ADDRESS(__prom_end, l7, g3) SPARC_LOAD_ADDRESS(__prom_end, l7, g3)
@ -389,35 +414,46 @@ prom_relocate_loop:
bne prom_relocate_loop bne prom_relocate_loop
inc 16,%g4 inc 16,%g4
#endif
! %o0 = stack pointer (relocated)
! %o1 = global data pointer (relocated)
! %o2 = text pointer (relocated)
! %g6 = relocation offset
! %l7 = _GLOBAL_OFFSET_TABLE_
/* Trap table has been moved, lets tell CPU about /* Trap table has been moved, lets tell CPU about
* the new trap table address * the new trap table address
*/ */
update_trap_table_address:
set CONFIG_SYS_RELOC_MONITOR_BASE, %g2 wr %g0, %o2, %tbr
wr %g0, %g2, %tbr
nop nop
nop nop
nop nop
/* Call relocated init functions */ update_stack_pointers:
jump: mov %o0, %fp
SPARC_LOAD_ADDRESS(cpu_init_f2, l7, o1) andn %fp, 0x0f, %fp ! align to 16 bytes
set CONFIG_SYS_RELOC_MONITOR_BASE,%o2 add %fp, -64, %fp ! make space for a window push
add %o1,%o2,%o1 mov %fp, %sp ! setup stack pointer
sub %o1,%g1,%o1
call %o1
clr %o0
SPARC_LOAD_ADDRESS(board_init_f, l7, o1) jump_board_init_r:
set CONFIG_SYS_RELOC_MONITOR_BASE,%o2 mov %o1, %o0 ! relocated global data pointer
SPARC_LOAD_ADDRESS(TEXT_START, l7, g1) mov %o2, %o1 ! relocated text pointer
add %o1,%o2,%o1 SPARC_LOAD_ADDRESS(board_init_r, l7, o3)
sub %o1,%g1,%o1 add %o3, %g6, %o3 ! add relocation offset
call %o1 call %o3
clr %o0 nop
dead: ta 0 ! if call returns... dead:
nop mov 1, %g1 ! For GRMON2 to exit normally.
ta 0 ! if call returns.. (unlikely)
nop
b dead ! infinte loop
nop
!------------------------------------------------------------------------------
/* Interrupt handler caller, /* Interrupt handler caller,
* reg L7: interrupt number * reg L7: interrupt number
@ -446,54 +482,56 @@ _irq_entry:
RESTORE_ALL RESTORE_ALL
!Window overflow trap handler. !------------------------------------------------------------------------------
/*
* Window overflow trap handler
*/
.global _window_overflow .global _window_overflow
_window_overflow: _window_overflow:
mov %wim, %l3 ! Calculate next WIM mov %wim, %l3 ! Calculate next WIM
mov %g1, %l7 mov %g1, %l7
srl %l3, 1, %g1 srl %l3, 1, %g1
sll %l3, (CONFIG_SYS_SPARC_NWINDOWS-1) , %l4 sll %l3, (CONFIG_SYS_SPARC_NWINDOWS-1), %l4
or %l4, %g1, %g1 or %g1, %l4, %g1
save ! Get into window to be saved. save ! Get into window to be saved.
mov %g1, %wim mov %g1, %wim
nop; nop; nop; nop
nop; st %l0, [%sp + 0] ! Save window to the stack
nop st %l1, [%sp + 4]
st %l0, [%sp + 0]; st %l2, [%sp + 8]
st %l1, [%sp + 4]; st %l3, [%sp + 12]
st %l2, [%sp + 8]; st %l4, [%sp + 16]
st %l3, [%sp + 12]; st %l5, [%sp + 20]
st %l4, [%sp + 16]; st %l6, [%sp + 24]
st %l5, [%sp + 20]; st %l7, [%sp + 28]
st %l6, [%sp + 24]; st %i0, [%sp + 32]
st %l7, [%sp + 28]; st %i1, [%sp + 36]
st %i0, [%sp + 32]; st %i2, [%sp + 40]
st %i1, [%sp + 36]; st %i3, [%sp + 44]
st %i2, [%sp + 40]; st %i4, [%sp + 48]
st %i3, [%sp + 44]; st %i5, [%sp + 52]
st %i4, [%sp + 48]; st %i6, [%sp + 56]
st %i5, [%sp + 52]; st %i7, [%sp + 60]
st %i6, [%sp + 56];
st %i7, [%sp + 60];
restore ! Go back to trap window. restore ! Go back to trap window.
mov %l7, %g1 mov %l7, %g1
jmp %l1 ! Re-execute save. jmp %l1 ! Re-execute save.
rett %l2 rett %l2
/* Window underflow trap handler. */
/*
* Window underflow trap handler
*/
.global _window_underflow .global _window_underflow
_window_underflow: _window_underflow:
mov %wim, %l3 ! Calculate next WIM mov %wim, %l3 ! Calculate next WIM
sll %l3, 1, %l4 srl %l3, (CONFIG_SYS_SPARC_NWINDOWS-1), %l5
srl %l3, (CONFIG_SYS_SPARC_NWINDOWS-1), %l5 sll %l3, 1, %l4
or %l5, %l4, %l5 or %l5, %l4, %l5
mov %l5, %wim mov %l5, %wim
nop; nop; nop nop; nop; nop
restore ! Two restores to get into the restore ! Two restores to get into the
restore ! window to restore restore ! window to restore
@ -516,9 +554,9 @@ _window_underflow:
save ! Get back to the trap window. save ! Get back to the trap window.
save save
jmp %l1 ! Re-execute restore. jmp %l1 ! Re-execute restore.
rett %l2 rett %l2
retl !------------------------------------------------------------------------------
_nmi_trap: _nmi_trap:
nop nop

View File

@ -14,6 +14,8 @@
#ifndef __CONFIG_H__ #ifndef __CONFIG_H__
#define __CONFIG_H__ #define __CONFIG_H__
#define CONFIG_SYS_GENERIC_BOARD
/* /*
* High Level Configuration Options * High Level Configuration Options
* (easy to change) * (easy to change)

View File

@ -15,6 +15,8 @@
#ifndef __CONFIG_H__ #ifndef __CONFIG_H__
#define __CONFIG_H__ #define __CONFIG_H__
#define CONFIG_SYS_GENERIC_BOARD
/* /*
* High Level Configuration Options * High Level Configuration Options
* (easy to change) * (easy to change)

View File

@ -13,6 +13,8 @@
#ifndef __CONFIG_H__ #ifndef __CONFIG_H__
#define __CONFIG_H__ #define __CONFIG_H__
#define CONFIG_SYS_GENERIC_BOARD
/* /*
* High Level Configuration Options * High Level Configuration Options
* (easy to change) * (easy to change)

View File

@ -13,6 +13,8 @@
#ifndef __CONFIG_H__ #ifndef __CONFIG_H__
#define __CONFIG_H__ #define __CONFIG_H__
#define CONFIG_SYS_GENERIC_BOARD
/* /*
* High Level Configuration Options * High Level Configuration Options
* (easy to change) * (easy to change)