2018-05-06 21:58:06 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
2012-06-05 11:33:17 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2012 Michael Walle
|
|
|
|
* Michael Walle <michael@walle.cc>
|
|
|
|
*
|
|
|
|
* Based on sheevaplug/sheevaplug.c by
|
|
|
|
* Marvell Semiconductor <www.marvell.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <common.h>
|
2020-05-10 17:40:00 +00:00
|
|
|
#include <bootstage.h>
|
2022-08-17 19:38:04 +00:00
|
|
|
#include <button.h>
|
2020-05-10 17:40:03 +00:00
|
|
|
#include <command.h>
|
2019-08-01 15:46:51 +00:00
|
|
|
#include <env.h>
|
2020-05-10 17:40:02 +00:00
|
|
|
#include <init.h>
|
2022-08-17 19:38:04 +00:00
|
|
|
#include <led.h>
|
|
|
|
#include <power/regulator.h>
|
2014-10-14 05:42:04 +00:00
|
|
|
#include <spi.h>
|
|
|
|
#include <spi_flash.h>
|
2012-06-05 11:33:17 +00:00
|
|
|
#include <asm/arch/cpu.h>
|
|
|
|
#include <asm/arch/mpp.h>
|
2020-10-31 03:38:53 +00:00
|
|
|
#include <asm/global_data.h>
|
2022-08-17 19:38:04 +00:00
|
|
|
#include <asm/io.h>
|
2020-05-10 17:40:11 +00:00
|
|
|
#include <linux/delay.h>
|
2012-06-05 11:33:17 +00:00
|
|
|
|
|
|
|
#include "lsxl.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Rescue mode
|
|
|
|
*
|
|
|
|
* Selected by holding the push button for 3 seconds, while powering on
|
|
|
|
* the device.
|
|
|
|
*
|
|
|
|
* These linkstations don't have a (populated) serial port. There is no
|
|
|
|
* way to access an (unmodified) board other than using the netconsole. If
|
|
|
|
* you want to recover from a bad environment setting or an empty environment,
|
|
|
|
* you can do this only with a working network connection. Therefore, a random
|
|
|
|
* ethernet address is generated if none is set and a DHCP request is sent.
|
|
|
|
* After a successful DHCP response is received, the network settings are
|
2012-10-04 06:54:25 +00:00
|
|
|
* configured and the ncip is unset. Therefore, all netconsole packets are
|
|
|
|
* broadcasted.
|
2012-06-05 11:33:17 +00:00
|
|
|
* Additionally, the bootsource is set to 'rescue'.
|
|
|
|
*/
|
|
|
|
|
|
|
|
DECLARE_GLOBAL_DATA_PTR;
|
|
|
|
|
2022-08-17 19:38:03 +00:00
|
|
|
static bool force_rescue_mode;
|
|
|
|
|
2012-06-05 11:33:17 +00:00
|
|
|
int board_early_init_f(void)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* default gpio configuration
|
|
|
|
* There are maximum 64 gpios controlled through 2 sets of registers
|
|
|
|
* the below configuration configures mainly initial LED status
|
|
|
|
*/
|
2014-10-22 10:13:11 +00:00
|
|
|
mvebu_config_gpio(LSXL_OE_VAL_LOW,
|
|
|
|
LSXL_OE_VAL_HIGH,
|
|
|
|
LSXL_OE_LOW, LSXL_OE_HIGH);
|
2012-06-05 11:33:17 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Multi-Purpose Pins Functionality configuration
|
|
|
|
* These strappings are taken from the original vendor uboot port.
|
|
|
|
*/
|
2012-11-26 11:27:36 +00:00
|
|
|
static const u32 kwmpp_config[] = {
|
2012-06-05 11:33:17 +00:00
|
|
|
MPP0_SPI_SCn,
|
|
|
|
MPP1_SPI_MOSI,
|
|
|
|
MPP2_SPI_SCK,
|
|
|
|
MPP3_SPI_MISO,
|
|
|
|
MPP4_UART0_RXD,
|
|
|
|
MPP5_UART0_TXD,
|
|
|
|
MPP6_SYSRST_OUTn,
|
|
|
|
MPP7_GPO,
|
|
|
|
MPP8_GPIO,
|
|
|
|
MPP9_GPIO,
|
|
|
|
MPP10_GPO, /* HDD power */
|
|
|
|
MPP11_GPIO, /* USB Vbus enable */
|
|
|
|
MPP12_SD_CLK,
|
|
|
|
MPP13_SD_CMD,
|
|
|
|
MPP14_SD_D0,
|
|
|
|
MPP15_SD_D1,
|
|
|
|
MPP16_SD_D2,
|
|
|
|
MPP17_SD_D3,
|
|
|
|
MPP18_GPO, /* fan speed high */
|
|
|
|
MPP19_GPO, /* fan speed low */
|
|
|
|
MPP20_GE1_0,
|
|
|
|
MPP21_GE1_1,
|
|
|
|
MPP22_GE1_2,
|
|
|
|
MPP23_GE1_3,
|
|
|
|
MPP24_GE1_4,
|
|
|
|
MPP25_GE1_5,
|
|
|
|
MPP26_GE1_6,
|
|
|
|
MPP27_GE1_7,
|
|
|
|
MPP28_GPIO,
|
|
|
|
MPP29_GPIO,
|
|
|
|
MPP30_GE1_10,
|
|
|
|
MPP31_GE1_11,
|
|
|
|
MPP32_GE1_12,
|
|
|
|
MPP33_GE1_13,
|
|
|
|
MPP34_GPIO,
|
|
|
|
MPP35_GPIO,
|
|
|
|
MPP36_GPIO, /* function LED */
|
|
|
|
MPP37_GPIO, /* alarm LED */
|
|
|
|
MPP38_GPIO, /* info LED */
|
|
|
|
MPP39_GPIO, /* power LED */
|
|
|
|
MPP40_GPIO, /* fan alarm */
|
|
|
|
MPP41_GPIO, /* funtion button */
|
|
|
|
MPP42_GPIO, /* power switch */
|
|
|
|
MPP43_GPIO, /* power auto switch */
|
|
|
|
MPP44_GPIO,
|
|
|
|
MPP45_GPIO,
|
|
|
|
MPP46_GPIO,
|
|
|
|
MPP47_GPIO,
|
|
|
|
MPP48_GPIO, /* function red LED */
|
|
|
|
MPP49_GPIO,
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
|
|
|
kirkwood_mpp_conf(kwmpp_config, NULL);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-08-17 19:38:04 +00:00
|
|
|
enum {
|
|
|
|
LSXL_LED_OFF,
|
|
|
|
LSXL_LED_ALARM,
|
|
|
|
LSXL_LED_POWER,
|
|
|
|
LSXL_LED_INFO,
|
|
|
|
};
|
2012-06-05 11:33:17 +00:00
|
|
|
|
2022-08-17 19:38:04 +00:00
|
|
|
static void __set_led(int alarm, int info, int power)
|
2012-06-05 11:33:17 +00:00
|
|
|
{
|
2022-08-17 19:38:04 +00:00
|
|
|
struct udevice *led;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = led_get_by_label("lsxl:red:alarm", &led);
|
|
|
|
if (!ret)
|
|
|
|
led_set_state(led, alarm);
|
|
|
|
ret = led_get_by_label("lsxl:amber:info", &led);
|
|
|
|
if (!ret)
|
|
|
|
led_set_state(led, info);
|
|
|
|
ret = led_get_by_label("lsxl:blue:power", &led);
|
|
|
|
if (!ret)
|
|
|
|
led_set_state(led, power);
|
2012-06-05 11:33:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void set_led(int state)
|
|
|
|
{
|
|
|
|
switch (state) {
|
2022-08-17 19:38:04 +00:00
|
|
|
case LSXL_LED_OFF:
|
|
|
|
__set_led(0, 0, 0);
|
2012-06-05 11:33:17 +00:00
|
|
|
break;
|
2022-08-17 19:38:04 +00:00
|
|
|
case LSXL_LED_ALARM:
|
|
|
|
__set_led(1, 0, 0);
|
2012-06-05 11:33:17 +00:00
|
|
|
break;
|
2022-08-17 19:38:04 +00:00
|
|
|
case LSXL_LED_INFO:
|
|
|
|
__set_led(0, 1, 0);
|
2012-06-05 11:33:17 +00:00
|
|
|
break;
|
2022-08-17 19:38:04 +00:00
|
|
|
case LSXL_LED_POWER:
|
|
|
|
__set_led(0, 0, 1);
|
2012-06-05 11:33:17 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int board_init(void)
|
|
|
|
{
|
|
|
|
/* address of boot parameters */
|
2014-10-22 10:13:13 +00:00
|
|
|
gd->bd->bi_boot_params = mvebu_sdram_bar(0) + 0x100;
|
2012-06-05 11:33:17 +00:00
|
|
|
|
2022-08-17 19:38:04 +00:00
|
|
|
set_led(LSXL_LED_POWER);
|
2012-06-05 11:33:17 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-07-30 10:47:12 +00:00
|
|
|
static void check_power_switch(void)
|
|
|
|
{
|
2022-08-17 19:38:04 +00:00
|
|
|
struct udevice *power_button, *hdd_power, *usb_power;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = button_get_by_label("Power-on Switch", &power_button);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
ret = regulator_get_by_platname("HDD Power", &hdd_power);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
ret = regulator_get_by_platname("USB Power", &usb_power);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
if (button_get_state(power_button) == BUTTON_OFF) {
|
|
|
|
ret = regulator_set_enable(hdd_power, false);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
ret = regulator_set_enable(usb_power, false);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
/* TODO: fan off */
|
|
|
|
set_led(LSXL_LED_OFF);
|
2012-07-30 10:47:12 +00:00
|
|
|
|
|
|
|
/* loop until released */
|
2022-08-17 19:38:04 +00:00
|
|
|
while (button_get_state(power_button) == BUTTON_OFF)
|
2012-07-30 10:47:12 +00:00
|
|
|
;
|
|
|
|
|
|
|
|
/* turn power on again */
|
2022-08-17 19:38:04 +00:00
|
|
|
ret = regulator_set_enable(hdd_power, true);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
ret = regulator_set_enable(usb_power, true);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
/* TODO: fan on */
|
|
|
|
set_led(LSXL_LED_POWER);
|
|
|
|
};
|
|
|
|
|
|
|
|
return;
|
|
|
|
err:
|
|
|
|
printf("error in %s\n", __func__);
|
2012-07-30 10:47:12 +00:00
|
|
|
}
|
|
|
|
|
2012-06-05 11:33:17 +00:00
|
|
|
void check_enetaddr(void)
|
|
|
|
{
|
|
|
|
uchar enetaddr[6];
|
|
|
|
|
2017-08-03 18:22:14 +00:00
|
|
|
if (!eth_env_get_enetaddr("ethaddr", enetaddr)) {
|
2012-06-05 11:33:17 +00:00
|
|
|
/* signal unset/invalid ethaddr to user */
|
2022-08-17 19:38:04 +00:00
|
|
|
set_led(LSXL_LED_INFO);
|
2012-06-05 11:33:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void erase_environment(void)
|
|
|
|
{
|
|
|
|
struct spi_flash *flash;
|
|
|
|
|
|
|
|
printf("Erasing environment..\n");
|
|
|
|
flash = spi_flash_probe(0, 0, 1000000, SPI_MODE_3);
|
|
|
|
if (!flash) {
|
|
|
|
printf("Erasing flash failed\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
spi_flash_erase(flash, CONFIG_ENV_OFFSET, CONFIG_ENV_SIZE);
|
|
|
|
spi_flash_free(flash);
|
|
|
|
do_reset(NULL, 0, 0, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void rescue_mode(void)
|
|
|
|
{
|
|
|
|
printf("Entering rescue mode..\n");
|
2017-08-03 18:22:09 +00:00
|
|
|
env_set("bootsource", "rescue");
|
2012-06-05 11:33:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void check_push_button(void)
|
|
|
|
{
|
2022-08-17 19:38:04 +00:00
|
|
|
struct udevice *func_button;
|
2012-06-05 11:33:17 +00:00
|
|
|
int i = 0;
|
|
|
|
|
2022-08-17 19:38:04 +00:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = button_get_by_label("Function Button", &func_button);
|
|
|
|
if (ret)
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
while (button_get_state(func_button) == BUTTON_ON) {
|
2012-06-05 11:33:17 +00:00
|
|
|
udelay(100000);
|
|
|
|
i++;
|
|
|
|
|
|
|
|
if (i == 10)
|
2022-08-17 19:38:04 +00:00
|
|
|
set_led(LSXL_LED_INFO);
|
2012-06-05 11:33:17 +00:00
|
|
|
|
|
|
|
if (i >= 100) {
|
2022-08-17 19:38:04 +00:00
|
|
|
set_led(LSXL_LED_ALARM);
|
2012-06-05 11:33:17 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i >= 100)
|
|
|
|
erase_environment();
|
|
|
|
else if (i >= 10)
|
2022-08-17 19:38:03 +00:00
|
|
|
force_rescue_mode = true;
|
2022-08-17 19:38:04 +00:00
|
|
|
|
|
|
|
return;
|
|
|
|
err:
|
|
|
|
printf("error in %s\n", __func__);
|
2022-08-17 19:38:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int board_early_init_r(void)
|
|
|
|
{
|
|
|
|
check_push_button();
|
|
|
|
|
|
|
|
return 0;
|
2012-06-05 11:33:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int misc_init_r(void)
|
|
|
|
{
|
2012-07-30 10:47:12 +00:00
|
|
|
check_power_switch();
|
2012-06-05 11:33:17 +00:00
|
|
|
check_enetaddr();
|
2022-08-17 19:38:03 +00:00
|
|
|
if (force_rescue_mode)
|
|
|
|
rescue_mode();
|
2012-06-05 11:33:17 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-05-03 20:48:58 +00:00
|
|
|
#if CONFIG_IS_ENABLED(BOOTSTAGE)
|
2012-06-05 11:33:17 +00:00
|
|
|
void show_boot_progress(int progress)
|
|
|
|
{
|
|
|
|
if (progress > 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* this is not an error, eg. bootp with autoload=no will trigger this */
|
|
|
|
if (progress == -BOOTSTAGE_ID_NET_LOADED)
|
|
|
|
return;
|
|
|
|
|
2022-08-17 19:38:04 +00:00
|
|
|
set_led(LSXL_LED_ALARM);
|
2012-06-05 11:33:17 +00:00
|
|
|
}
|
|
|
|
#endif
|