blob: d7ceb4d660c61f63ac76eb84f77d8bc4219a5768 [file] [log] [blame] [edit]
/*
* Copyright (C) 2015 Amlogic, Inc. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include <common.h>
#include <asm/io.h>
#include <malloc.h>
#include <errno.h>
#include <environment.h>
#include <fdt_support.h>
#include <linux/libfdt.h>
#include <amlogic/cpu_id.h>
#include <asm/arch/secure_apb.h>
#include <asm/arch/pinctrl_init.h>
#include <linux/sizes.h>
#include <asm-generic/gpio.h>
#include <dm.h>
#include <asm/armv8/mmu.h>
#include <amlogic/aml_v3_burning.h>
#include <amlogic/aml_v2_burning.h>
#include <linux/mtd/partitions.h>
#include <asm/arch/bl31_apis.h>
#include <amlogic/aml_mtd.h>
#include <asm/arch/stick_mem.h>
#ifdef CONFIG_AML_VPU
#include <amlogic/media/vpu/vpu.h>
#endif
#ifdef CONFIG_AML_VPP
#include <amlogic/media/vpp/vpp.h>
#endif
#ifdef CONFIG_AML_VOUT
#include <amlogic/media/vout/aml_vout.h>
#endif
#ifdef CONFIG_AML_HDMITX20
#include <amlogic/media/vout/hdmitx/hdmitx_module.h>
#endif
#ifdef CONFIG_AML_LCD
#include <amlogic/media/vout/lcd/lcd_vout.h>
#endif
#ifdef CONFIG_RX_RTERM
#include <amlogic/aml_hdmirx.h>
#endif
DECLARE_GLOBAL_DATA_PTR;
void sys_led_init(void)
{
}
int serial_set_pin_port(unsigned long port_base)
{
return 0;
}
int dram_init(void)
{
gd->ram_size = PHYS_SDRAM_1_SIZE;
return 0;
}
/* secondary_boot_func
* this function should be write with asm, here, is is only for compiling pass
* */
void secondary_boot_func(void)
{
}
int board_eth_init(bd_t *bis)
{
return 0;
}
int active_clk(void)
{
struct udevice *clk = NULL;
int err;
err = uclass_get_device_by_name(UCLASS_CLK,
"xtal-clk", &clk);
if (err) {
pr_err("Can't find xtal-clk clock (%d)\n", err);
return err;
}
err = uclass_get_device_by_name(UCLASS_CLK,
"clock-controller@0", &clk);
if (err) {
pr_err("Can't find clock-controller@0 clock (%d)\n", err);
return err;
}
return 0;
}
#ifdef CONFIG_AML_HDMITX20
static void hdmitx_set_hdmi_5v(void)
{
/*Power on VCC_5V for HDMI_5V*/
}
#endif
void board_init_mem(void) {
#if 1
/* config bootm low size, make sure whole dram/psram space can be used */
phys_size_t ram_size;
char *env_tmp;
env_tmp = env_get("bootm_size");
if (!env_tmp) {
ram_size = (readl(SYSCTRL_SEC_STATUS_REG4) & ~0xfffffUL) << 4;
if (ram_size >= 0xe0000000)
ram_size = 0xe0000000;
env_set_hex("bootm_low", 0);
env_set_hex("bootm_size", ram_size);
}
#endif
}
int board_init(void)
{
printf("board init\n");
#ifdef CONFIG_PXP_EMULATOR
return 0;
#else
/* The non-secure watchdog is enabled in BL2 TEE, disable it */
run_command("watchdog off", 0);
printf("watchdog disable\n");
aml_set_bootsequence(0);
//Please keep try usb boot first in board_init, as other init before usb may cause burning failure
#if defined(CONFIG_AML_V3_FACTORY_BURN) && defined(CONFIG_AML_V3_USB_TOOl)
if ((0x1b8ec003 != readl(SYSCTRL_SEC_STICKY_REG2)) && (0x1b8ec004 != readl(SYSCTRL_SEC_STICKY_REG2)))
{ aml_v3_factory_usb_burning(0, gd->bd); }
#endif//#if defined(CONFIG_AML_V3_FACTORY_BURN) && defined(CONFIG_AML_V3_USB_TOOl)
#ifdef CONFIG_RTK_USB_BT
clrbits_le32(PADCTRL_GPIOC_OEN, (0x1 << 11));
clrbits_le32(PADCTRL_GPIOC_O, (0x1 << 11));
setbits_le32(PADCTRL_GPIOC_O, (0x1 << 11));
#endif
#if 0
active_clk();
#ifdef CONFIG_AML_HDMITX20
hdmitx_set_hdmi_5v();
hdmitx_init();
#endif
#endif
pinctrl_devices_active(PIN_CONTROLLER_NUM);
/*set vcc5V*/
return 0;
#endif
}
int board_late_init(void)
{
char outputModePre[30] = {};
#ifdef CONFIG_PXP_EMULATOR
char outputModeCur[30] = {};
#endif
printf("board late init\n");
get_stick_reboot_flag();
/* ****************************************************
* 1.setup bootup resource
* ****************************************************
*/
//default uboot env need before anyone use it
if (env_get("default_env")) {
printf("factory reset, need default all uboot env.\n");
run_command("defenv_reserv; setenv upgrade_step 2; saveenv;", 0);
}
run_command("echo upgrade_step $upgrade_step; if itest ${upgrade_step} == 1; then "\
"defenv_reserv; setenv upgrade_step 2; saveenv; fi;", 0);
board_init_mem();
run_command("run bcb_cmd", 0);
#ifndef CONFIG_SYSTEM_RTOS //prue rtos not need dtb
if ( run_command("run common_dtb_load", 0) ) {
printf("Fail in load dtb with cmd[%s]\n", env_get("common_dtb_load"));
} else {
//load dtb here then users can directly use 'fdt' command
run_command("if fdt addr ${dtb_mem_addr}; then else echo no valid dtb at ${dtb_mem_addr};fi;", 0);
}
#endif//#ifndef CONFIG_SYSTEM_RTOS //prue rtos not need dtb
/* ****************************************************
* 2.use bootup resource after setup
* ****************************************************
*/
if (env_get("outputmode"))
strncpy(outputModePre, env_get("outputmode"), 29);
#ifdef CONFIG_AML_FACTORY_BURN_LOCAL_UPGRADE //try auto upgrade from ext-sdcard
aml_try_factory_sdcard_burning(0, gd->bd);
#endif//#ifdef CONFIG_AML_FACTORY_BURN_LOCAL_UPGRADE
//auto enter usb mode after board_late_init if 'adnl.exe setvar burnsteps 0x1b8ec003'
#if defined(CONFIG_AML_V3_FACTORY_BURN) && defined(CONFIG_AML_V3_USB_TOOl)
if (0x1b8ec003 == readl(SYSCTRL_SEC_STICKY_REG2))
{ aml_v3_factory_usb_burning(0, gd->bd); }
#endif//#if defined(CONFIG_AML_V3_FACTORY_BURN) && defined(CONFIG_AML_V3_USB_TOOl)
/* load unifykey */
run_command("keyman init 0x1234", 0);
#ifdef CONFIG_PXP_EMULATOR
#ifdef CONFIG_AML_VPU
vpu_probe();
#endif
#ifdef CONFIG_AML_VPP
vpp_init();
#endif
#ifdef CONFIG_RX_RTERM
rx_set_phy_rterm();
#endif
run_command("ini_model", 0);
#ifdef CONFIG_AML_VOUT
vout_probe();
#endif
#ifdef CONFIG_AML_LCD
lcd_probe();
#endif
run_command("amlsecurecheck", 0);
run_command("update_tries", 0);
if (env_get("outputmode")) {
strncpy(outputModeCur, env_get("outputmode"), 29);
}
if (strcmp(outputModeCur,outputModePre)) {
printf("outputMode changed, old:%s - new:%s\n", outputModePre, outputModeCur);
run_command("update_env_part -p outputmode", 0);
}
unsigned char chipid[16];
memset(chipid, 0, 16);
if (get_chip_id(chipid, 16) != -1) {
char chipid_str[32];
int i, j;
char buf_tmp[4];
memset(chipid_str, 0, 32);
char *buff = &chipid_str[0];
for (i = 0, j = 0; i < 12; ++i) {
sprintf(&buf_tmp[0], "%02x", chipid[15 - i]);
if (strcmp(buf_tmp, "00") != 0) {
sprintf(buff + j, "%02x", chipid[15 - i]);
j = j + 2;
}
}
env_set("cpu_id", chipid_str);
printf("buff: %s\n", buff);
} else {
env_set("cpu_id", "1234567890");
}
#endif
return 0;
}
phys_size_t get_effective_memsize(void)
{
phys_size_t ddr_size;
// >>16 -> MB, <<20 -> real size, so >>16<<20 = <<4
#if defined(CONFIG_SYS_MEM_TOP_HIDE)
ddr_size = (readl(SYSCTRL_SEC_STATUS_REG4) & ~0xfffffUL) << 4;
if (ddr_size >= 0xe0000000)
ddr_size = 0xe0000000;
return (ddr_size - CONFIG_SYS_MEM_TOP_HIDE);
#else
ddr_size = (readl(SYSCTRL_SEC_STATUS_REG4) & ~0xfffffUL) << 4;
if (ddr_size >= 0xe0000000)
ddr_size = 0xe0000000;
return ddr_size;
#endif /* CONFIG_SYS_MEM_TOP_HIDE */
}
phys_size_t get_ddr_info_size(void)
{
phys_size_t ddr_size = (((readl(SYSCTRL_SEC_STATUS_REG4)) & ~0xffffUL) << 4);
return ddr_size;
}
ulong board_get_usable_ram_top(ulong total_size)
{
unsigned long top = gd->ram_top;
if (top >= 0xE0000000UL)
return 0xE0000000UL;
return top;
}
static struct mm_region bd_mem_map[] = {
{
.virt = 0x00000000UL,
.phys = 0x00000000UL,
.size = 0x80000000UL,
.attrs = PTE_BLOCK_MEMTYPE(MT_NORMAL) |
PTE_BLOCK_INNER_SHARE
}, {
.virt = 0xe0000000UL,
.phys = 0xe0000000UL,
.size = 0x20000000UL,
.attrs = PTE_BLOCK_MEMTYPE(MT_DEVICE_NGNRNE) |
PTE_BLOCK_NON_SHARE |
PTE_BLOCK_PXN | PTE_BLOCK_UXN
}, {
/* List terminator */
0,
}
};
struct mm_region *mem_map = bd_mem_map;
int mach_cpu_init(void) {
//printf("\nmach_cpu_init\n");
unsigned long nddrSize = (readl(SYSCTRL_SEC_STATUS_REG4) & ~0xfffffUL) << 4;
if ( nddrSize <= 0xe0000000 )
bd_mem_map[0].size = nddrSize;
return 0;
}
int ft_board_setup(void *blob, bd_t *bd)
{
/* eg: bl31/32 rsv */
return 0;
}
/* partition table for spinor flash */
#ifdef CONFIG_SPI_FLASH
static const struct mtd_partition spiflash_partitions[] = {
{
.name = "env",
.offset = 0,
.size = 1 * SZ_256K,
},
{
.name = "dtb",
.offset = 0,
.size = 1 * SZ_256K,
},
{
.name = "boot",
.offset = 0,
.size = 2 * SZ_1M,
},
/* last partition get the rest capacity */
{
.name = "user",
.offset = MTDPART_OFS_APPEND,
.size = MTDPART_SIZ_FULL,
}
};
const struct mtd_partition *get_spiflash_partition_table(int *partitions)
{
*partitions = ARRAY_SIZE(spiflash_partitions);
return spiflash_partitions;
}
#endif /* CONFIG_SPI_FLASH */
#ifdef CONFIG_MESON_NFC
static struct mtd_partition normal_partition_info[] = {
{
.name = BOOT_BL2E,
.offset = 0,
.size = 0,
},
{
.name = BOOT_BL2X,
.offset = 0,
.size = 0,
},
{
.name = BOOT_DDRFIP,
.offset = 0,
.size = 0,
},
{
.name = BOOT_DEVFIP,
.offset = 0,
.size = 0,
},
{
.name = "logo",
.offset = 0,
.size = 2*SZ_1M,
},
{
.name = "recovery",
.offset = 0,
.size = 16*SZ_1M,
},
{
.name = "boot",
.offset = 0,
.size = 16*SZ_1M,
},
{
.name = "system",
.offset = 0,
.size = 64*SZ_1M,
},
/* last partition get the rest capacity */
{
.name = "data",
.offset = MTDPART_OFS_APPEND,
.size = MTDPART_SIZ_FULL,
},
};
struct mtd_partition *get_aml_mtd_partition(void)
{
return normal_partition_info;
}
int get_aml_partition_count(void)
{
return ARRAY_SIZE(normal_partition_info);
}
#endif
/* partition table */
/* partition table for spinand flash */
#if (defined(CONFIG_SPI_NAND) || defined(CONFIG_MTD_SPI_NAND))
static const struct mtd_partition spinand_partitions[] = {
{
.name = "logo",
.offset = 0,
.size = 2 * SZ_1M,
},
{
.name = "recovery",
.offset = 0,
.size = 16 * SZ_1M,
},
{
.name = "boot",
.offset = 0,
.size = 16 * SZ_1M,
},
{
.name = "system",
.offset = 0,
.size = 64 * SZ_1M,
},
/* last partition get the rest capacity */
{
.name = "data",
.offset = MTDPART_OFS_APPEND,
.size = MTDPART_SIZ_FULL,
}
};
const struct mtd_partition *get_spinand_partition_table(int *partitions)
{
*partitions = ARRAY_SIZE(spinand_partitions);
return spinand_partitions;
}
#endif /* CONFIG_SPI_NAND */
#ifdef CONFIG_MULTI_DTB
int checkhw(char * name)
{
char loc_name[64] = {0};
unsigned long ddr_size = 0;
int i;
cpu_id_t cpu_id = get_cpu_id();
for (i = 0; i < CONFIG_NR_DRAM_BANKS; i++)
ddr_size += gd->bd->bi_dram[i].size;
#if defined(CONFIG_SYS_MEM_TOP_HIDE)
ddr_size += CONFIG_SYS_MEM_TOP_HIDE;
#endif
switch (ddr_size) {
case 0x80000000:
if (cpu_id.chip_rev == 0xA)
strcpy(loc_name, "t5m-reva_t963d4_ay309-2g\0");
else if (cpu_id.chip_rev == 0xB)
strcpy(loc_name, "t5m_t963d4_ay309-2g\0");
break;
case 0xc0000000:
if (cpu_id.chip_rev == 0xA)
strcpy(loc_name, "t5m-reva_t963d4_ay309-3g\0");
else if (cpu_id.chip_rev == 0xB)
strcpy(loc_name, "t5m_t963d4_ay309-3g\0");
break;
default:
strcpy(loc_name, "t5m_t963d4_unsupport");
break;
}
/* set aml_dt */
strcpy(name, loc_name);
env_set("aml_dt", loc_name);
return 0;
}
#endif
const char * const _env_args_reserve_[] =
{
"lock",
"upgrade_step",
"bootloader_version",
"model_name",
"hdmimode",
"outputmode",
NULL//Keep NULL be last to tell END
};
int __attribute__((weak)) mmc_initialize(bd_t *bis){ return 0;}
int __attribute__((weak)) do_bootm(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[]){ return 0;}
void __attribute__((weak)) set_working_fdt_addr(ulong addr) {}
int __attribute__((weak)) ofnode_read_u32_default(ofnode node, const char *propname, u32 def) {return 0;}
void __attribute__((weak)) md5_wd (unsigned char *input, int len, unsigned char output[16], unsigned int chunk_sz){}