mirror of
https://github.com/smaeul/u-boot.git
synced 2025-10-24 09:38:18 +01:00
254 lines
5.9 KiB
ArmAsm
254 lines
5.9 KiB
ArmAsm
/* This is the memory initialization functions, the function
|
|
* implemented below initializes each memory controller
|
|
* found and specified by the input grlib_mctrl_handler structure.
|
|
*
|
|
* After the memory controllers have been initialized the stack
|
|
* can be used.
|
|
*
|
|
* (C) Copyright 2010, 2015
|
|
* Daniel Hellstrom, Cobham Gaisler, daniel@gaisler.com.
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0+
|
|
*/
|
|
|
|
#include <ambapp.h>
|
|
#include "memcfg.h"
|
|
#include <config.h>
|
|
|
|
.seg "text"
|
|
.globl _nomem_memory_ctrl_init
|
|
.globl _nomem_mctrl_init, _nomem_ahbmctrl_init
|
|
.extern _nomem_find_apb
|
|
.extern _nomem_find_ahb
|
|
|
|
|
|
/* FUNCTION
|
|
* _nomem_memory_controller_init(struct grlib_mctrl_handler *mem_handlers)
|
|
*
|
|
* Initialize AMBA devices, _nomem_amba_init() has prepared i0-i5
|
|
* with the AHB buses on the system.
|
|
*
|
|
* For each entry in mem_handlers find the VENDOR:DEVICE and handle it
|
|
* by calling the handler function pointer.
|
|
*
|
|
* Constraints:
|
|
* i6, i7, o6, l7, l6, g3, g4, g5, g6, g7 is used by caller
|
|
* o7 is return address
|
|
* l5 reserved for this function for future use.
|
|
*
|
|
* Arguments
|
|
* - o0 Pointer to memory handler array
|
|
*
|
|
* Results
|
|
* - o0 Number of memory controllers found
|
|
*
|
|
* Clobbered
|
|
* - o0 (Current AHB slave conf address)
|
|
* - l0 (mem handler entry address)
|
|
* - l1 (Return value, number of memory controllers found)
|
|
* - o7 (function pointer)
|
|
* - l0, l1, l2, l3, l4, g1, g2 (used by _nomem_ambapp_find_buses)
|
|
* - o0, o1, o2, o3, o4, o5 (Used as arguments)
|
|
*
|
|
* - g1 ( level 1 return address)
|
|
* - g2 ( level 2 return address)
|
|
*/
|
|
|
|
_nomem_memory_ctrl_init:
|
|
/* At this point all AHB buses has been found and the I/O Areas of
|
|
* all AHB buses is stored in the i0-i5 registers. Max 6 buses. Next,
|
|
* memory controllers are found by searching all buses for matching
|
|
* VENDOR:DEVICE. The VENDOR:DEVICE to search for are taken from the
|
|
* mem_handlers array. For each match the function pointer stored in
|
|
* the mem_handler entry is called to handle the hardware setup.
|
|
*/
|
|
mov %o7, %g1 /* Save return address */
|
|
mov %o0, %l0
|
|
mov %g0, %l1 /* The return value */
|
|
|
|
.L_do_one_mem_handler:
|
|
ld [%l0 + MH_FUNC], %o7
|
|
cmp %o7, %g0
|
|
be .L_all_mctrl_handled
|
|
nop
|
|
|
|
/*** Scan for memory controller ***/
|
|
|
|
/* Set up argments, o5 not used by _nomem_find_apb */
|
|
ldub [%l0 + MH_TYPE], %o5
|
|
clr %o4
|
|
clr %o3
|
|
ldub [%l0 + MH_INDEX], %o2
|
|
ld [%l0 + MH_VENDOR_DEVICE], %o1
|
|
|
|
/* An empty config? */
|
|
cmp %o5, DEV_NONE
|
|
beq .L_all_mctrl_next
|
|
|
|
/* Select function (APB or AHB) */
|
|
cmp %o5, DEV_APB_SLV
|
|
bne .L_find_ahb_memctrl
|
|
clr %o0
|
|
.L_find_apb_memctrl:
|
|
call _nomem_find_apb /* Scan for APB slave device */
|
|
nop
|
|
|
|
/* o3 = iobar address
|
|
* o4 = AHB Bus index
|
|
*
|
|
* REG ADR = ((iobar >> 12) & (iobar << 4) & 0xfff00) | "APB Base"
|
|
*/
|
|
ld [%o3 + AMBA_APB_IOBAR_OFS], %o5
|
|
srl %o5, 12, %o2
|
|
sll %o5, 4, %o5
|
|
and %o2, %o5, %o5
|
|
set 0xfff00, %o2
|
|
and %o2, %o5, %o5
|
|
sethi %hi(0xfff00000), %o2
|
|
and %o3, %o2, %o2
|
|
or %o5, %o2, %o5 /* Register base address */
|
|
|
|
ba .L_call_one_mem_handler
|
|
nop
|
|
|
|
.L_find_ahb_memctrl:
|
|
call _nomem_find_ahb /* Scan for AHB Slave or Master.
|
|
* o5 determine type. */
|
|
nop
|
|
clr %o5
|
|
|
|
/* Call the handler function if the hardware was found
|
|
*
|
|
* o0 = mem_handler
|
|
* o1 = Configuration address
|
|
* o2 = AHB Bus index
|
|
* o3 = APB Base register (if APB Slave)
|
|
*
|
|
* Constraints:
|
|
* i0-i7, l0, l1, l5, g1, g3-g7 may no be used.
|
|
*/
|
|
.L_call_one_mem_handler:
|
|
cmp %o0, %g0
|
|
be .L_all_mctrl_next
|
|
mov %l0, %o0 /* Mem handler pointer */
|
|
mov %o3, %o1 /* AMBA PnP Configuration address */
|
|
mov %o4, %o2 /* AHB Bus index */
|
|
ld [%l0 + MH_FUNC], %o7 /* Get Function pointer */
|
|
call %o7
|
|
mov %o5, %o3 /* APB Register Base Address */
|
|
|
|
inc %l1 /* Number of Memory controllers
|
|
* handled. */
|
|
|
|
/* Do next entry in mem_handlers */
|
|
.L_all_mctrl_next:
|
|
ba .L_do_one_mem_handler
|
|
add %l0, MH_STRUCT_SIZE, %l0
|
|
|
|
.L_all_mctrl_handled:
|
|
mov %g1, %o7 /* Restore return address */
|
|
retl
|
|
mov %l1, %o0
|
|
|
|
|
|
|
|
/* Generic Memory controller initialization routine (APB Registers)
|
|
*
|
|
* o0 = mem_handler structure pointer
|
|
* o1 = Configuration address
|
|
* o2 = AHB Bus index
|
|
* o3 = APB Base register
|
|
*
|
|
* Clobbered
|
|
* o0-o4
|
|
*/
|
|
_nomem_mctrl_init:
|
|
ld [%o0 + MH_PRIV], %o0 /* Get Private structure */
|
|
ld [%o0], %o1 /* Get Reg Mask */
|
|
and %o1, 0xff, %o1
|
|
add %o0, REGS_OFS, %o0 /* Point to first reg */
|
|
.L_do_one_reg:
|
|
andcc %o1, 0x1, %g0
|
|
beq .L_do_next_reg
|
|
ld [%o0], %o2
|
|
ld [%o3], %o4
|
|
and %o4, %o2, %o4
|
|
ld [%o0 + 4], %o2
|
|
or %o4, %o2, %o4
|
|
st %o4, [%o3]
|
|
|
|
.L_do_next_reg:
|
|
add %o0, REGS_SIZE, %o0
|
|
add %o3, 4, %o3
|
|
srl %o1, 1, %o1
|
|
cmp %o1, 0
|
|
bne .L_do_one_reg
|
|
nop
|
|
|
|
/* No more registers to write */
|
|
retl
|
|
nop
|
|
|
|
|
|
|
|
/* Generic Memory controller initialization routine (AHB Registers)
|
|
*
|
|
* o0 = mem_handler structure pointer
|
|
* o1 = Configuration address of memory controller
|
|
* o2 = AHB Bus index
|
|
*
|
|
* Clobbered
|
|
* o0-o5
|
|
*/
|
|
_nomem_ahbmctrl_init:
|
|
ld [%o0 + MH_PRIV], %o0 /* Get Private structure */
|
|
|
|
/* Get index of AHB MBAR to get registers from */
|
|
ld [%o0], %o5
|
|
add %o0, 4, %o0
|
|
|
|
/* Get Address of MBAR in PnP info */
|
|
add %o5, 4, %o5
|
|
sll %o5, 2, %o5
|
|
add %o5, %o1, %o5 /* Address of MBAR */
|
|
|
|
/* Get Address of registers from PnP information
|
|
* Address is in AHB I/O format, i.e. relative to bus
|
|
*
|
|
* ADR = (iobar & (iobar << 16) & 0xfff00000)
|
|
* IOADR = (ADR >> 12) | "APB Base"
|
|
*/
|
|
ld [%o5], %o5
|
|
sll %o5, 16, %o4
|
|
and %o5, %o4, %o5
|
|
sethi %hi(0xfff00000), %o4
|
|
and %o5, %o4, %o5 /* ADR */
|
|
and %o4, %o1, %o4
|
|
srl %o5, 12, %o5
|
|
or %o5, %o4, %o3 /* IOADR in o3 */
|
|
|
|
ld [%o0], %o1 /* Get Reg Mask */
|
|
and %o1, 0xff, %o1
|
|
add %o0, REGS_OFS, %o0 /* Point to first reg */
|
|
.L_do_one_ahbreg:
|
|
andcc %o1, 0x1, %g0
|
|
beq .L_do_next_reg
|
|
ld [%o0], %o2
|
|
ld [%o3], %o4
|
|
and %o4, %o2, %o4
|
|
ld [%o0 + 4], %o2
|
|
or %o4, %o2, %o4
|
|
st %o4, [%o3]
|
|
|
|
.L_do_next_ahbreg:
|
|
add %o0, REGS_SIZE, %o0
|
|
add %o3, 4, %o3
|
|
srl %o1, 1, %o1
|
|
cmp %o1, 0
|
|
bne .L_do_one_reg
|
|
nop
|
|
|
|
/* No more registers to write */
|
|
retl
|
|
nop
|