mirror of
https://github.com/AsahiLinux/u-boot
synced 2024-11-12 16:07:30 +00:00
7b8657e2bd
Make sure value in register r0 and r1 is preserved and passed to the board_init_ll() and mxs_common_spl_init() where it can be processed further. The value in r0 can be configured during the BootStream generation to arbitary value, r1 contains pointer to return value from CALL'd function. This patch also clears the value in r0 before returning to BootROM to make sure the BootROM is not confused by this value. Finally, this patch cleans up some comments in the start.S file. Signed-off-by: Marek Vasut <marex@denx.de> Cc: Fabio Estevam <fabio.estevam@freescale.com> Cc: Stefano Babic <sbabic@denx.de>
135 lines
4.3 KiB
C
135 lines
4.3 KiB
C
/*
|
|
* Freescale i.MX28 Boot setup
|
|
*
|
|
* Copyright (C) 2011 Marek Vasut <marek.vasut@gmail.com>
|
|
* on behalf of DENX Software Engineering GmbH
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0+
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <config.h>
|
|
#include <asm/io.h>
|
|
#include <asm/arch/imx-regs.h>
|
|
#include <asm/arch/sys_proto.h>
|
|
#include <asm/gpio.h>
|
|
|
|
#include "mxs_init.h"
|
|
|
|
/*
|
|
* This delay function is intended to be used only in early stage of boot, where
|
|
* clock are not set up yet. The timer used here is reset on every boot and
|
|
* takes a few seconds to roll. The boot doesn't take that long, so to keep the
|
|
* code simple, it doesn't take rolling into consideration.
|
|
*/
|
|
void early_delay(int delay)
|
|
{
|
|
struct mxs_digctl_regs *digctl_regs =
|
|
(struct mxs_digctl_regs *)MXS_DIGCTL_BASE;
|
|
|
|
uint32_t st = readl(&digctl_regs->hw_digctl_microseconds);
|
|
st += delay;
|
|
while (st > readl(&digctl_regs->hw_digctl_microseconds))
|
|
;
|
|
}
|
|
|
|
#define MUX_CONFIG_BOOTMODE_PAD (MXS_PAD_3V3 | MXS_PAD_4MA | MXS_PAD_NOPULL)
|
|
static const iomux_cfg_t iomux_boot[] = {
|
|
#if defined(CONFIG_MX23)
|
|
MX23_PAD_LCD_D00__GPIO_1_0 | MUX_CONFIG_BOOTMODE_PAD,
|
|
MX23_PAD_LCD_D01__GPIO_1_1 | MUX_CONFIG_BOOTMODE_PAD,
|
|
MX23_PAD_LCD_D02__GPIO_1_2 | MUX_CONFIG_BOOTMODE_PAD,
|
|
MX23_PAD_LCD_D03__GPIO_1_3 | MUX_CONFIG_BOOTMODE_PAD,
|
|
MX23_PAD_LCD_D04__GPIO_1_4 | MUX_CONFIG_BOOTMODE_PAD,
|
|
MX23_PAD_LCD_D05__GPIO_1_5 | MUX_CONFIG_BOOTMODE_PAD,
|
|
#elif defined(CONFIG_MX28)
|
|
MX28_PAD_LCD_D00__GPIO_1_0 | MUX_CONFIG_BOOTMODE_PAD,
|
|
MX28_PAD_LCD_D01__GPIO_1_1 | MUX_CONFIG_BOOTMODE_PAD,
|
|
MX28_PAD_LCD_D02__GPIO_1_2 | MUX_CONFIG_BOOTMODE_PAD,
|
|
MX28_PAD_LCD_D03__GPIO_1_3 | MUX_CONFIG_BOOTMODE_PAD,
|
|
MX28_PAD_LCD_D04__GPIO_1_4 | MUX_CONFIG_BOOTMODE_PAD,
|
|
MX28_PAD_LCD_D05__GPIO_1_5 | MUX_CONFIG_BOOTMODE_PAD,
|
|
#endif
|
|
};
|
|
|
|
static uint8_t mxs_get_bootmode_index(void)
|
|
{
|
|
uint8_t bootmode = 0;
|
|
int i;
|
|
uint8_t masked;
|
|
|
|
/* Setup IOMUX of bootmode pads to GPIO */
|
|
mxs_iomux_setup_multiple_pads(iomux_boot, ARRAY_SIZE(iomux_boot));
|
|
|
|
#if defined(CONFIG_MX23)
|
|
/* Setup bootmode pins as GPIO input */
|
|
gpio_direction_input(MX23_PAD_LCD_D00__GPIO_1_0);
|
|
gpio_direction_input(MX23_PAD_LCD_D01__GPIO_1_1);
|
|
gpio_direction_input(MX23_PAD_LCD_D02__GPIO_1_2);
|
|
gpio_direction_input(MX23_PAD_LCD_D03__GPIO_1_3);
|
|
gpio_direction_input(MX23_PAD_LCD_D05__GPIO_1_5);
|
|
|
|
/* Read bootmode pads */
|
|
bootmode |= (gpio_get_value(MX23_PAD_LCD_D00__GPIO_1_0) ? 1 : 0) << 0;
|
|
bootmode |= (gpio_get_value(MX23_PAD_LCD_D01__GPIO_1_1) ? 1 : 0) << 1;
|
|
bootmode |= (gpio_get_value(MX23_PAD_LCD_D02__GPIO_1_2) ? 1 : 0) << 2;
|
|
bootmode |= (gpio_get_value(MX23_PAD_LCD_D03__GPIO_1_3) ? 1 : 0) << 3;
|
|
bootmode |= (gpio_get_value(MX23_PAD_LCD_D05__GPIO_1_5) ? 1 : 0) << 5;
|
|
#elif defined(CONFIG_MX28)
|
|
/* Setup bootmode pins as GPIO input */
|
|
gpio_direction_input(MX28_PAD_LCD_D00__GPIO_1_0);
|
|
gpio_direction_input(MX28_PAD_LCD_D01__GPIO_1_1);
|
|
gpio_direction_input(MX28_PAD_LCD_D02__GPIO_1_2);
|
|
gpio_direction_input(MX28_PAD_LCD_D03__GPIO_1_3);
|
|
gpio_direction_input(MX28_PAD_LCD_D04__GPIO_1_4);
|
|
gpio_direction_input(MX28_PAD_LCD_D05__GPIO_1_5);
|
|
|
|
/* Read bootmode pads */
|
|
bootmode |= (gpio_get_value(MX28_PAD_LCD_D00__GPIO_1_0) ? 1 : 0) << 0;
|
|
bootmode |= (gpio_get_value(MX28_PAD_LCD_D01__GPIO_1_1) ? 1 : 0) << 1;
|
|
bootmode |= (gpio_get_value(MX28_PAD_LCD_D02__GPIO_1_2) ? 1 : 0) << 2;
|
|
bootmode |= (gpio_get_value(MX28_PAD_LCD_D03__GPIO_1_3) ? 1 : 0) << 3;
|
|
bootmode |= (gpio_get_value(MX28_PAD_LCD_D04__GPIO_1_4) ? 1 : 0) << 4;
|
|
bootmode |= (gpio_get_value(MX28_PAD_LCD_D05__GPIO_1_5) ? 1 : 0) << 5;
|
|
#endif
|
|
|
|
for (i = 0; i < ARRAY_SIZE(mxs_boot_modes); i++) {
|
|
masked = bootmode & mxs_boot_modes[i].boot_mask;
|
|
if (masked == mxs_boot_modes[i].boot_pads)
|
|
break;
|
|
}
|
|
|
|
return i;
|
|
}
|
|
|
|
void mxs_common_spl_init(const uint32_t arg, const uint32_t *resptr,
|
|
const iomux_cfg_t *iomux_setup,
|
|
const unsigned int iomux_size)
|
|
{
|
|
struct mxs_spl_data *data = (struct mxs_spl_data *)
|
|
((CONFIG_SYS_TEXT_BASE - sizeof(struct mxs_spl_data)) & ~0xf);
|
|
uint8_t bootmode = mxs_get_bootmode_index();
|
|
|
|
mxs_iomux_setup_multiple_pads(iomux_setup, iomux_size);
|
|
mxs_power_init();
|
|
|
|
mxs_mem_init();
|
|
data->mem_dram_size = mxs_mem_get_size();
|
|
|
|
data->boot_mode_idx = bootmode;
|
|
|
|
mxs_power_wait_pswitch();
|
|
}
|
|
|
|
/* Support aparatus */
|
|
inline void board_init_f(unsigned long bootflag)
|
|
{
|
|
for (;;)
|
|
;
|
|
}
|
|
|
|
inline void board_init_r(gd_t *id, ulong dest_addr)
|
|
{
|
|
for (;;)
|
|
;
|
|
}
|