@@ -13,12 +13,20 @@ config SYS_CONFIG_NAME
default "sifive-fu540"
config SYS_TEXT_BASE
+ default 0x80200000 if SPL
default 0x80000000 if !RISCV_SMODE
default 0x80200000 if RISCV_SMODE
+config SPL_TEXT_BASE
+ default 0x08000000
+
+config SPL_OPENSBI_LOAD_ADDR
+ default 0x80000000
+
config BOARD_SPECIFIC_OPTIONS # dummy
def_bool y
select GENERIC_RISCV
+ select SUPPORT_SPL
imply CMD_DHCP
imply CMD_EXT2
imply CMD_EXT4
@@ -3,3 +3,7 @@
# Copyright (c) 2019 Western Digital Corporation or its affiliates.
obj-y += fu540.o
+
+ifdef CONFIG_SPL_BUILD
+obj-y += spl.o
+endif
new file mode 100644
@@ -0,0 +1,33 @@
+/* SPDX-License-Identifier: GPL-2.0+ */
+/*
+ * Copyright (c) 2019 SiFive, Inc
+ */
+
+#ifndef FU540_MEMORY_MAP
+#define FU540_MEMORY_MAP
+
+#include <asm/arch/gpio.h>
+#include "ux00prci.h"
+
+/****************************************************************************
+ * Platform definitions
+ *****************************************************************************/
+
+/* Memory map */
+#define GPIO_CTRL_ADDR _AC(0x10060000, UL)
+
+#define PHYSICAL_FILTER_CTRL_ADDR _AC(0x100b8000, UL)
+
+#define UX00DDR_CTRL_ADDR _AC(0x100b0000, UL)
+#define UX00PRCI_CTRL_ADDR _AC(0x10000000, UL)
+
+/* Helper functions */
+#define _REG32(p, i) (*(volatile uint32_t *)((p) + (i)))
+
+#define UX00PRCI_REG(offset) \
+ _REG32(UX00PRCI_CTRL_ADDR, \
+ offset)
+
+#define GPIO_REG(offset) _REG32(GPIO_CTRL_ADDR, offset)
+
+#endif /* FU540_MEMORY_MAP */
@@ -11,6 +11,7 @@
#include <linux/delay.h>
#include <linux/io.h>
#include <misc.h>
+#include <spl.h>
/*
* This define is a value used for error/unknown serial.
@@ -114,3 +115,26 @@ int board_init(void)
return 0;
}
+
+#ifdef CONFIG_SPL
+void board_boot_order(u32 *spl_boot_list)
+{
+ u8 i;
+ u32 boot_devices[] = {
+#ifdef CONFIG_SPL_MMC_SUPPORT
+ BOOT_DEVICE_MMC1,
+#endif
+ };
+
+ for (i = 0; i < ARRAY_SIZE(boot_devices); i++)
+ spl_boot_list[i] = boot_devices[i];
+}
+#endif
+
+#ifdef CONFIG_SPL_LOAD_FIT
+int board_fit_config_name_match(const char *name)
+{
+ /* boot using first FIT config */
+ return 0;
+}
+#endif
new file mode 100644
@@ -0,0 +1,78 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Copyright (c) 2019 SiFive, Inc
+ *
+ * Authors:
+ * Pragnesh Patel <pragnesh.patel at sifive.com>
+ */
+
+#include <common.h>
+#include <spl.h>
+#include <misc.h>
+#include <dm.h>
+
+#include "fu540-memory-map.h"
+
+#define DDRCTLPLL_F 55
+#define DDRCTLPLL_Q 2
+
+#define PHY_NRESET 0x1000
+
+long nsec_per_cyc = 300; /* 33.333MHz */
+void nsleep(long nsec)
+{
+ long step = nsec_per_cyc * 2;
+
+ while (nsec > 0)
+ nsec -= step;
+}
+
+void init_clk_and_ddr(void)
+{
+ int ret;
+ struct udevice *dev;
+
+ /* PRCI init */
+ ret = uclass_get_device(UCLASS_CLK, 0, &dev);
+ if (ret) {
+ debug("Clock init failed: %d\n", ret);
+ return;
+ }
+
+ ret = uclass_get_device(UCLASS_RAM, 0, &dev);
+ if (ret) {
+ printf("DRAM init failed: %d\n", ret);
+ return;
+ }
+
+ /*
+ * GEMGXL init VSC8541 PHY reset sequence;
+ * leave pull-down active for 2ms
+ */
+ nsleep(2000000);
+ /* Set GPIO 12 (PHY NRESET) to OE=1 and OVAL=1 */
+ GPIO_REG(GPIO_OUTPUT_VAL) |= PHY_NRESET;
+ GPIO_REG(GPIO_OUTPUT_EN) |= PHY_NRESET;
+ nsleep(100);
+
+ /* Reset PHY again to enter unmanaged mode */
+ GPIO_REG(GPIO_OUTPUT_VAL) &= ~PHY_NRESET;
+ nsleep(100);
+ GPIO_REG(GPIO_OUTPUT_VAL) |= PHY_NRESET;
+ nsleep(15000000);
+}
+
+void board_init_f(ulong dummy)
+{
+ int ret;
+
+ ret = spl_early_init();
+ if (ret)
+ panic("spl_early_init() failed: %d\n", ret);
+
+ arch_cpu_init_dm();
+
+ init_clk_and_ddr();
+
+ preloader_console_init();
+}
new file mode 100644
@@ -0,0 +1,56 @@
+/* SPDX-License-Identifier: GPL-2.0+ */
+/*
+ * Copyright (c) 2019 SiFive, Inc
+ */
+
+#ifndef _SIFIVE_UX00PRCI_H
+#define _SIFIVE_UX00PRCI_H
+
+/* Register offsets */
+#define UX00PRCI_HFROSCCFG (0x0000)
+#define UX00PRCI_COREPLLCFG (0x0004)
+#define UX00PRCI_COREPLLOUT (0x0008)
+#define UX00PRCI_DDRPLLCFG (0x000C)
+#define UX00PRCI_DDRPLLOUT (0x0010)
+#define UX00PRCI_GEMGXLPLLCFG (0x001C)
+#define UX00PRCI_GEMGXLPLLOUT (0x0020)
+#define UX00PRCI_CORECLKSELREG (0x0024)
+#define UX00PRCI_DEVICESRESETREG (0x0028)
+#define UX00PRCI_CLKMUXSTATUSREG (0x002C)
+#define UX00PRCI_PROCMONCFG (0x00F0)
+
+#define PLL_R(x) (((x) & 0x3F) << 0)
+#define PLL_F(x) (((x) & 0x1FF) << 6)
+#define PLL_Q(x) (((x) & 0x7) << 15)
+#define PLL_RANGE(x) (((x) & 0x7) << 18)
+#define PLL_BYPASS(x) (((x) & 0x1) << 24)
+#define PLL_FSE(x) (((x) & 0x1) << 25)
+#define PLL_LOCK(x) (((x) & 0x1) << 31)
+
+#define PLLOUT_DIV(x) (((x) & 0x7F) << 0)
+#define PLLOUT_DIV_BY_1(x) (((x) & 0x1) << 8)
+#define PLLOUT_CLK_EN(x) (((x) & 0x1) << 31)
+
+#define PLL_R_default 0x1
+#define PLL_F_default 0x1F
+#define PLL_Q_default 0x3
+#define PLL_RANGE_default 0x0
+#define PLL_BYPASS_default 0x1
+#define PLL_FSE_default 0x1
+
+#define PLLOUT_DIV_default 0x0
+#define PLLOUT_DIV_BY_1_default 0x0
+#define PLLOUT_CLK_EN_default 0x0
+
+#define PLL_CORECLKSEL_HFXIN 0x1
+#define PLL_CORECLKSEL_COREPLL 0x0
+
+#define DEVICESRESET_DDR_CTRL_RST_N(x) (((x) & 0x1) << 0)
+#define DEVICESRESET_DDR_AXI_RST_N(x) (((x) & 0x1) << 1)
+#define DEVICESRESET_DDR_AHB_RST_N(x) (((x) & 0x1) << 2)
+#define DEVICESRESET_DDR_PHY_RST_N(x) (((x) & 0x1) << 3)
+#define DEVICESRESET_GEMGXL_RST_N(x) (((x) & 0x1) << 5)
+
+#define CLKMUX_STATUS_TLCLKSEL (0x1 << 1)
+
+#endif // _SIFIVE_UX00PRCI_H
@@ -11,6 +11,22 @@
#include <linux/sizes.h>
+#ifdef CONFIG_SPL
+
+#define CONFIG_SPL_MAX_SIZE 0x00100000
+#define CONFIG_SPL_BSS_START_ADDR 0x85000000
+#define CONFIG_SPL_BSS_MAX_SIZE 0x00100000
+#define CONFIG_SYS_SPL_MALLOC_START (CONFIG_SPL_BSS_START_ADDR + \
+ CONFIG_SPL_BSS_MAX_SIZE)
+#define CONFIG_SYS_SPL_MALLOC_SIZE 0x00100000
+
+#define CONFIG_SPL_LOAD_FIT_ADDRESS 0x84000000
+
+#define CONFIG_SPL_STACK (0x08000000 + 0x001D0000 - \
+ GENERATED_GBL_DATA_SIZE)
+
+#endif
+
#define CONFIG_SYS_SDRAM_BASE 0x80000000
#define CONFIG_SYS_INIT_SP_ADDR (CONFIG_SYS_SDRAM_BASE + SZ_2M)
@@ -24,6 +40,7 @@
/* Environment options */
+#ifndef CONFIG_SPL_BUILD
#define BOOT_TARGET_DEVICES(func) \
func(MMC, mmc, 0) \
func(DHCP, dhcp, na)
@@ -43,5 +60,6 @@
#define CONFIG_PREBOOT \
"setenv fdt_addr ${fdtcontroladdr};" \
"fdt addr ${fdtcontroladdr};"
+#endif
#endif /* __CONFIG_H */