// SPDX-License-Identifier: GPL-2.0+ /* * (C) Copyright 2012 Henrik Nordstrom * * (C) Copyright 2007-2011 * Allwinner Technology Co., Ltd. * Tom Cubie * * Some init for sunxi platform. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include static int gpio_init(void) { return 0; } void s_init(void) { #if !defined(CONFIG_ARM_CORTEX_CPU_IS_UP) && !defined(CONFIG_ARM64) /* Enable SMP mode for CPU0, by setting bit 6 of Auxiliary Ctl reg */ asm volatile( "mrc p15, 0, r0, c1, c0, 1\n" "orr r0, r0, #1 << 6\n" "mcr p15, 0, r0, c1, c0, 1\n" ::: "r0"); #endif clock_init(); timer_init(); gpio_init(); eth_init_board(); } /* The sunxi internal brom will try to loader external bootloader * from mmc0, nand flash, mmc2. */ uint32_t sunxi_get_boot_device(void) { int boot_source; /* * When booting from the SD card or NAND memory, the "eGON.BT0" * signature is expected to be found in memory at the address 0x0004 * (see the "mksunxiboot" tool, which generates this header). * * When booting in the FEL mode over USB, this signature is patched in * memory and replaced with something else by the 'fel' tool. This other * signature is selected in such a way, that it can't be present in a * valid bootable SD card image (because the BROM would refuse to * execute the SPL in this case). * * This checks for the signature and if it is not found returns to * the FEL code in the BROM to wait and receive the main u-boot * binary over USB. If it is found, it determines where SPL was * read from. */ if (!is_boot0_magic(SPL_ADDR + 4)) /* eGON.BT0 */ return BOOT_DEVICE_BOARD; boot_source = readb(SPL_ADDR + 0x28); switch (boot_source) { case SUNXI_BOOTED_FROM_MMC0: return BOOT_DEVICE_MMC1; case SUNXI_BOOTED_FROM_NAND: return BOOT_DEVICE_NAND; case SUNXI_BOOTED_FROM_MMC2: return BOOT_DEVICE_MMC2; case SUNXI_BOOTED_FROM_SPI: return BOOT_DEVICE_SPI; } panic("Unknown boot source %d\n", boot_source); return -1; /* Never reached */ } void reset_cpu(ulong addr) { #if defined(CONFIG_SUNXI_WDT_V2) static const struct sunxi_wdog *wdog = (struct sunxi_wdog *)SUNXI_WDT_BASE; #if defined(CONFIG_MACH_SUN50IW12) /*wait deinit done, so we wont freeze on reboot*/ mdelay(500); #endif writel(((WDT_CFG_KEY << 16) | 0x02), SUNXI_WDT_BASE + 0x18); /* Set the watchdog for its shortest interval (.5s) and wait */ writel(((WDT_CFG_KEY << 16) | WDT_MODE_EN), &wdog->srst); while (1) { } #elif defined(CONFIG_SUNXI_GEN_SUN4I) || defined(CONFIG_MACH_SUN8I_R40) static const struct sunxi_wdog *wdog = &((struct sunxi_timer_reg *)SUNXI_TIMER_BASE)->wdog; /* Set the watchdog for its shortest interval (.5s) and wait */ writel(WDT_MODE_RESET_EN | WDT_MODE_EN, &wdog->mode); writel(WDT_CTRL_KEY | WDT_CTRL_RESTART, &wdog->ctl); while (1) { /* sun5i sometimes gets stuck without this */ writel(WDT_MODE_RESET_EN | WDT_MODE_EN, &wdog->mode); } #elif defined(CONFIG_SUNXI_GEN_SUN6I) static const struct sunxi_wdog *wdog = ((struct sunxi_timer_reg *)SUNXI_TIMER_BASE)->wdog; /* Set the watchdog for its shortest interval (.5s) and wait */ writel(WDT_CFG_RESET, &wdog->cfg); writel(WDT_MODE_EN, &wdog->mode); writel(WDT_CTRL_KEY | WDT_CTRL_RESTART, &wdog->ctl); while (1) { } #elif defined(CONFIG_SUNXI_NCAT) || defined(CONFIG_SUNXI_NCAT_V2) static const struct sunxi_wdog *wdog = ((struct sunxi_timer_reg *)SUNXI_TIMER_BASE)->wdog; /* Set the watchdog for its shortest interval (.5s) and wait */ writel(WDT_CFG_RESET, &wdog->cfg); writel(WDT_MODE_EN, &wdog->mode); while (1) { } #elif defined(CONFIG_SUNXI_VERSION1) struct sunxi_timer_reg *timer_reg = (struct sunxi_timer_reg *)SUNXI_TIMER_BASE; struct sunxi_wdog *wdog = &timer_reg->wdog[0]; /* enable watchdog */ wdog->mode |= 3; while (1) { } #endif } void * board_fdt_blob_setup(void) { /* check u-boot with dtb.bin*/ if (fdt_check_header(&_end) == 0) return (void *)&_end; /* uboot base + 2M offset */ return (void*)(CONFIG_SYS_TEXT_BASE + SUNXI_DTB_OFFSET); } #if !CONFIG_IS_ENABLED(SYS_DCACHE_OFF) && !defined(CONFIG_ARM64) void enable_caches(void) { /* Enable D-cache. I-cache is already enabled in start.S */ dcache_enable(); } #endif