/** * @file hv_boot_Main.c * @brief boot main source code. * @details This file provides the following functions: \n * (1) Initialization and de-initialization functions \n * (2) Start and stop functions \n * (3) Feed functions \n * * @author HiView SoC Software Team * @version 1.0.0 * @date 2022-08-18 * @copyright Copyright(c),2022-8, Hiview Software. All rights reserved. * @par History: * *
Author Date Change Description *
HiView SoC Software Team 2022-08-23 init *
*/ #include #include #include "hv_chip_Config.h" #include "hv_chip_Memory.h" #include "hv_comm_FlashConfig.h" #include "hv_boot_Lzss.h" #include "hv_boot_Common.h" __attribute__((long_call)) extern int main(int argc, char **argv); __attribute__((long_call)) extern void Hv_Vos_Init(void); #define PAGEMASK_256M 0x1FFFE000 #define EHINV (1 << 10) #define ENTRY_MASK (7) // GLOBAL, VALID, DIRTY #define ENTRY_MASK_INVALID (1) // GLOBAL, VALID, DIRTY #define FLASH_END 0x4000000 //#define CONFIG_4LINE_DMA #define CONFIG_USE_DMA /* temp ddr address for code to decompress. */ #define DDR_TEMP_ADDR 0x82000000 unsigned int g_uiFlashUpgradeOffset = 0; /* TLB memory mapping PAddr - VAddr - kseg0 - uncached kseg0 ----------------------------------------------------- 0x00000000 - N/A - 0x80000000 - 0xA0000000 0x10000000 - N/A - 0x90000000 - 0xB0000000 0x80000000 - 0x60000000 - N/A - N/A 0x90000000 - 0x70000000 - N/A - N/A */ static void Hv_Boot_InitTlb(void) { unsigned int uiVaddr = 0; unsigned int uiEvenAddr = 0; unsigned int uiOddAaddr = 0; uiVaddr = 0x00000000; mips_tlbrwr2(uiVaddr | EHINV, ENTRY_MASK_INVALID, ENTRY_MASK_INVALID, PAGEMASK_256M); uiVaddr = 0x20000000; mips_tlbrwr2(uiVaddr | EHINV, ENTRY_MASK_INVALID, ENTRY_MASK_INVALID, PAGEMASK_256M); uiVaddr = 0x30000000; mips_tlbrwr2(uiVaddr | EHINV, ENTRY_MASK_INVALID, ENTRY_MASK_INVALID, PAGEMASK_256M); /* mapping 0x80000000 => 0x60000000 mapping 0x90000000 => 0x70000000 */ uiVaddr = 0x60000000; uiEvenAddr = 0x80000000; uiOddAaddr = 0x90000000; mips_tlbrwr2(uiVaddr | EHINV, (uiEvenAddr >> 12 << 6) | ENTRY_MASK, (uiOddAaddr >> 12 << 6) | ENTRY_MASK, PAGEMASK_256M); return; } void Hv_Boot_ClearBss(void) { extern void *__bss_start, *__bss_end; unsigned int *puiStart = (unsigned int*)(&__bss_start); unsigned int *puiEnd = (unsigned int*)(&__bss_end); while ((unsigned int)puiStart < (unsigned int)puiEnd) { *puiStart ++ = 0; } mips_flush_cache(); return; } void Hv_Boot_CopyDdrCodeData(void) { extern void *_text_lma,*__text_start, *__rodata_end; extern void *_data_lma, *__data_start, *__data_end; unsigned int iIndex = 0; unsigned int uiDataSize = 0; unsigned int *puiData = NULL; unsigned int *puiFlash = NULL; /* Copy text and rodata segment from Flash to ddr */ uiDataSize = ((unsigned int)(&__rodata_end) - (unsigned int)(&__text_start)) / sizeof(unsigned int); puiData = (unsigned int *)&__text_start; puiFlash = ((unsigned int *)&_text_lma) + g_uiFlashUpgradeOffset; for (iIndex = 0; iIndex < uiDataSize ; iIndex++) { puiData[iIndex] = puiFlash[iIndex]; } /* Copy data segment from Flash to ddr */ uiDataSize = ((unsigned int)(&__data_end) - (unsigned int)(&__data_start)) / sizeof(unsigned int); puiData = (unsigned int *)&__data_start; puiFlash = ((unsigned int *)&_data_lma) + g_uiFlashUpgradeOffset; for (iIndex = 0; iIndex < uiDataSize ; iIndex++) { puiData[iIndex] = puiFlash[iIndex]; } return; } static void Hv_Boot_CopyDdrCodeDataCompressed(int uiCompressed) { unsigned int uiDdrAddr = 0; extern void *_text_lma, *__text_start, *__data_end; unsigned int *puiData = NULL; unsigned int *puiFlash = NULL; unsigned int iIndex = 0; unsigned int uiDataSize = 0; uiDataSize = ((unsigned int)(&__data_end) - (unsigned int)(&__text_start)) / sizeof(unsigned int); puiData = (unsigned int *)&__text_start; puiFlash = ((unsigned int *)&_text_lma) + g_uiFlashUpgradeOffset; if (uiCompressed) { puiData = (unsigned int *)DDR_TEMP_ADDR; uiDataSize = HV_FLASH_CONFIG_CODE_PART_SIZE / sizeof(unsigned int); } Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_INFO, ">>> Hv_Boot_CopyDdrCodeDataCompressed enter.\n"); Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_DEBUG, uiDataSize); Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_DEBUG, (unsigned int)puiData); Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_DEBUG, (unsigned int)puiFlash); for (iIndex = 0; iIndex < uiDataSize ; iIndex++) { puiData[iIndex] = puiFlash[iIndex]; } Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_DEBUG, ">>> Hv_Boot_CopyDdrCodeDataCompressed over.\n"); return; } extern void Hv_Ddr_Init(void); void Hv_Boot_InitDDR(void) { Hv_Ddr_Init(); return; } static void _Boot_DmaCopyFlashToDDR(unsigned int uiFlashAddr, unsigned int uiDdrAddr, unsigned int uiDataSize) { /*use two dma channel for rx/tx,most high speed is 4 division*/ unsigned int auiCmdAddr[2] = {0}; unsigned int uiPageCount = 0; unsigned int uiSurplusSize = 0; unsigned int iIndex = 0; Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_INFO, ">>> Hv_Boot_DmaCopyFlashToDDR.\n"); Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_DEBUG, uiDataSize); Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_DEBUG, uiDdrAddr); Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_DEBUG, uiFlashAddr); BOOT_WT32(CPU_SYS_DMAC_CTL0_L, 0xa09405); BOOT_WT32(CPU_SYS_DMAC_CFG0_L, 0x800000); BOOT_WT32(CPU_SYS_DMAC_CFG0_H, 0x82); BOOT_WT32(CPU_SYS_DMAC_MASKTFR, 0x101); BOOT_WT32(CPU_SYS_DMAC_CLEARTFR, 0x01); BOOT_WT32(CPU_SYS_DMACFGREG, 0x01); BOOT_W32_FIELD(SYS_REG_SH5_TOP0_REG_SH5_WDG_FILED2, reg_sh5_qspi_xip_en, 0x0); BOOT_WT32(CPU_SYS_2_QSPI_SSIENR, 0x00); BOOT_RD32(CPU_SYS_2_QSPI_ICR); #ifdef CONFIG_4LINE_DMA /* 8bit read only, four line select */ BOOT_WT32(CPU_SYS_2_QSPI_CTRLR0, 0x800807); #else BOOT_WT32(CPU_SYS_2_QSPI_CTRLR0, 0x400807); #endif /* division set */ BOOT_WT32(CPU_SYS_2_QSPI_BAUDR, 6); BOOT_WT32(CPU_SYS_2_QSPI_TXFTLR, 0x08); BOOT_WT32(CPU_SYS_2_QSPI_RXFTLR, 0x00); /* addr 24bit, instru 8bit, standard mode. */ BOOT_W32_FIELD(CPU_SYS_2_QSPI_SPI_CTRLR0, TRANS_TYPE, 0); BOOT_W32_FIELD(CPU_SYS_2_QSPI_SPI_CTRLR0, ADDR_L, 0x6); BOOT_W32_FIELD(CPU_SYS_2_QSPI_SPI_CTRLR0, INST_L, 0x2); /* read cmd 4line */ #ifdef CONFIG_4LINE_DMA auiCmdAddr[0]= 0x6B; #else /* read cmd 2line */ auiCmdAddr[0]= 0x3B; #endif /* read dma enable */ BOOT_WT32(CPU_SYS_2_QSPI_DMACR, 0x01); BOOT_WT32(CPU_SYS_2_QSPI_DMARDLR, 0x07); Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_DEBUG, ">>> Read time start.\n"); uiPageCount = uiDataSize >> 11; uiSurplusSize = uiDataSize & 0x7ff; Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_DEBUG, uiPageCount); Hv_Boot_PrintHex(BOOT_DEBUG_LEVEL_DEBUG, uiSurplusSize); for (iIndex = 0; iIndex < uiPageCount; iIndex++) { BOOT_WT32(CPU_SYS_DMAC_SAR0, CPU_SYS_2_QSPI_DR0); BOOT_WT32(CPU_SYS_DMAC_DAR0, uiDdrAddr & 0x1fffffff); BOOT_WT32(CPU_SYS_DMAC_CTL0_H, 0x800); BOOT_WT32(CPU_SYS_DMAC_CHENREG, 0x101); BOOT_WT32(CPU_SYS_2_QSPI_CTRLR1, 0x7ff); BOOT_WT32(CPU_SYS_2_QSPI_SSIENR, 0x01); BOOT_WT32(CPU_SYS_2_QSPI_DR0, auiCmdAddr[0]); BOOT_WT32(CPU_SYS_2_QSPI_DR0, uiFlashAddr); while(!(BOOT_RD32(CPU_SYS_DMAC_STATUSINT) & 0x01)); BOOT_WT32(CPU_SYS_DMAC_CLEARTFR, 0x1); Hv_Boot_Wait_Us(1); while((BOOT_RD32(CPU_SYS_2_QSPI_SR) & 0x01) == 0x01); Hv_Boot_Wait_Us(1); BOOT_WT32(CPU_SYS_2_QSPI_SSIENR,0x0); Hv_Boot_Wait_Us(1); uiFlashAddr = uiFlashAddr + 0x800; uiDdrAddr = uiDdrAddr + 0x800; } if (uiSurplusSize > 0) { BOOT_WT32(CPU_SYS_DMAC_SAR0, CPU_SYS_2_QSPI_DR0); BOOT_WT32(CPU_SYS_DMAC_DAR0, uiDdrAddr & 0x1fffffff); BOOT_WT32(CPU_SYS_DMAC_CTL0_H, uiSurplusSize); BOOT_WT32(CPU_SYS_DMAC_CHENREG, 0x101); BOOT_WT32(CPU_SYS_2_QSPI_CTRLR1, uiSurplusSize-1); BOOT_WT32(CPU_SYS_2_QSPI_SSIENR, 0x01); BOOT_WT32(CPU_SYS_2_QSPI_DR0, auiCmdAddr[0]); BOOT_WT32(CPU_SYS_2_QSPI_DR0, uiFlashAddr); while(!(BOOT_RD32(CPU_SYS_DMAC_STATUSINT) & 0x01)); BOOT_WT32(CPU_SYS_DMAC_CLEARTFR, 0x1); while((BOOT_RD32(CPU_SYS_2_QSPI_SR) & 0x01) == 0x01); } BOOT_WT32(CPU_SYS_DMAC_MASKTFR,0x100); BOOT_WT32(CPU_SYS_DMACFGREG,0x0); Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_DEBUG, ">>> Read time end.\n"); /* togle ssien before xip mode. */ BOOT_WT32(CPU_SYS_2_QSPI_SSIENR, 0x00); //BOOT_RD32(CPU_SYS_2_QSPI_ICR); //BOOT_WT32(CPU_SYS_2_QSPI_BAUDR, 0x8); //BOOT_WT32(CPU_SYS_2_QSPI_DMACR, 0x00); BOOT_WT32(CPU_SYS_2_QSPI_SSIENR, 0x1); BOOT_W32_FIELD(SYS_REG_SH5_TOP0_REG_SH5_WDG_FILED2, reg_sh5_qspi_xip_en, 0x1); return; } static void Hv_Boot_DmaCopyFlashToDDR(int uiCompressed) { unsigned int uiDdrAddr = 0; unsigned int uiFlashAddr = 0; extern void *_text_lma, *__text_start, *__data_end; unsigned int iIndex = 0; unsigned int uiDataSize = 0; uiDataSize = ((unsigned int)(&__data_end) - (unsigned int)(&__text_start)); uiDdrAddr = (unsigned int )&__text_start; uiFlashAddr = (unsigned int)((unsigned int)&_text_lma - 0xBfc00000) + g_uiFlashUpgradeOffset; if (uiCompressed) { uiDdrAddr = DDR_TEMP_ADDR; uiDataSize = HV_FLASH_CONFIG_CODE_PART_SIZE; } if (uiFlashAddr + uiDataSize > FLASH_END) { uiDataSize = FLASH_END - uiFlashAddr; } _Boot_DmaCopyFlashToDDR(uiFlashAddr, uiDdrAddr, uiDataSize); return; } static void Hv_Boot_CopyPqFlashToDDR(void) { unsigned int uiDdrAddr = DDR_TEMP_ADDR; unsigned int uiFlashAddr = HV_FLASH_CONFIG_PQ_DATA_PART_START; unsigned int uiDataSize = HV_FLASH_CONFIG_PQ_DATA_PART_SIZE; if (uiFlashAddr + uiDataSize > FLASH_END) { uiDataSize = FLASH_END - uiFlashAddr; } _Boot_DmaCopyFlashToDDR(uiFlashAddr, uiDdrAddr, uiDataSize); return; } static void Hv_Boot_CopyLogoFlashToDDR(void) { unsigned int uiDdrAddr = HV_MEMORY_CONFIG_OSD_LOGO_DDR_START; unsigned int uiFlashAddr = HV_FLASH_CONFIG_LOGO_PART_START; unsigned int uiDataSize = HV_FLASH_CONFIG_LOGO_PART_SIZE; if (uiFlashAddr + uiDataSize > FLASH_END) { uiDataSize = FLASH_END - uiFlashAddr; } _Boot_DmaCopyFlashToDDR(uiFlashAddr, uiDdrAddr, uiDataSize); return; } void Hv_Boot_WatchdogInit(void) { BOOT_WT32(SYS_REG_SH5_TOP0_REG_SH5_WDG_FILED5, 5000); /* wdt reload */ BOOT_W32_FIELD(SYS_REG_SH5_TOP0_REG_SH5_WDG_FILED7, reg_sh5_wdg_reload_en, 1); BOOT_W32_FIELD(SYS_REG_SH5_TOP0_REG_SH5_WDG_FILED7, reg_sh5_wdg_reload_en, 0); unsigned int uiVal = BOOT_RD32(SYS_REG_SH5_TOP0_REG_SH5_WDG_FILED4); /* count enable */ uiVal |= 0x00000001; /* wdg reset enable to reset whole system */ uiVal |= 0x00000008; /* set count period 1ms from APB_CLK */ BOOT_WT32(SYS_REG_SH5_TOP0_REG_SH5_WDG_FILED6, (HV_CHIP_CLK_APB / 1000)); /* start the wdt */ BOOT_WT32(SYS_REG_SH5_TOP0_REG_SH5_WDG_FILED4, uiVal); } static unsigned char s_ucBootType = 0; static unsigned char s_cAcFlag = 0; void Hv_Boot_Check_BootReason(void) { int iData = 0; iData = BOOT_RD32(STANDBY_STANDBY_MCU_REG_STBY_MCU6); if (4 == (iData&0xff))/* wakeup */ { iData = BOOT_RD32(STANDBY_STANDBY_MCU_REG_STBY_MCU7); s_ucBootType = (iData&0xff); BOOT_WT32(SW_DUMMY_JTAG, s_ucBootType); } return; } unsigned char Hv_Boot_Get_BootReason(void) { return s_ucBootType; } unsigned char Hv_Boot_Get_BootType(void) { return s_cAcFlag; } static void BootPwrOff(void) { /* E_MAILBOX_CMD_RISC_TO_PM_NORMAL */ BOOT_WT32(STANDBY_STANDBY_MCU_REG_STBY_MCU10, 7); BOOT_WT32(STANDBY_STANDBY_MCU_REG_STBY_MCU15, 1); Hv_Boot_Wait_Ms(20); /* E_MAILBOX_CMD_RISC_TO_PM_POWER_OFF_ACK */ BOOT_WT32(STANDBY_STANDBY_MCU_REG_STBY_MCU10, 5); BOOT_WT32(STANDBY_STANDBY_MCU_REG_STBY_MCU15, 1); Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_WAR, ">>> poweroff\n"); while(1); } void Hv_Boot_Check_SecondPowerOn(void) { unsigned int uiData = 0; unsigned int uiLoop = 0; unsigned int uiAddr = HV_FLASH_CONFIG_START_XIP+HV_FLASH_CONFIG_MONITOR_DATA_PART_START+HV_FLASH_CONFIG_TIMER_START; BOOT_R32_DECL_VAR(); s_cAcFlag = BOOT_R32_FIELD(STANDBY_AON_REG_PMU_REG3, reg_first_start_flag); if (s_cAcFlag) { BOOT_W32_FIELD(STANDBY_AON_REG_PMU_REG3, reg_first_start_flag, 0); #if (HV_PROJECT_CONFIG_AC_POWER_STATE == 1) BootPwrOff(); #elif (HV_PROJECT_CONFIG_AC_POWER_STATE == 2) return; #else ; #endif } else { return; } for (uiLoop = 0; uiLoop<4096; uiLoop+=16) { /* magic TIMD or HT7315 */ if ((BOOT_RD32(uiAddr+uiLoop+4) != 0x54494D45) && (BOOT_RD32(uiAddr+uiLoop+4) != 0x48547315)) { break; } uiData = BOOT_RD32(uiAddr+uiLoop+0xc); } if (8 == (uiData&0xff))/* poweroff */ { BootPwrOff(); } return; } int _start(void) { int iRet = 0; char cGpioIndex = 16; int iPqCompressSize = 0; char *pucAddr = NULL; #ifdef SW_DUMMY_DEBUG while(HV_RD32(SW_DUMMY_JTAG)==0x11223344); #endif mips_bissr(SR_CU1); Hv_Chip_DebugUartInit(115200); Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_WAR, ">>> BOOT RISC0\n"); /* find upgrade section. */ if (*(char *)(HV_FLASH_CONFIG_START_XIP + HV_FLASH_CONFIG_PM_DATA_START + 4 + 5) == 0xaa) { Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_WAR, ">>> user second part.\n"); //g_uiFlashUpgradeOffset = HV_FLASH_CONFIG_BACK_SIZE; } #ifdef CONFIG_USER_WATCHDOG_ON Hv_Boot_WatchdogInit(); #endif Hv_Boot_Check_BootReason(); Hv_Boot_Check_SecondPowerOn(); if (11 != s_ucBootType) { #if HV_BOARD_CONFIG_LED_R_POL BOOT_W32_FIELD(STANDBY_STANDBY_PERI_REG_PWM0, reg_pwm_pol, 0); #else BOOT_W32_FIELD(STANDBY_STANDBY_PERI_REG_PWM0, reg_pwm_pol, 1); #endif BOOT_W32_FIELD(STANDBY_STANDBY_PERI_REG_PWM0, reg_pwm_ch_en, 0); BOOT_W32_FIELD(STANDBY_STANDBY_PERI_REG_PWM0, reg_pwm_output_mode, 1); #if (HV_BOARD_CONFIG_GPIO_LED_B == 61) /* pin61-gpio21-bit16*/ BOOT_W32_FIELD(RX_SH08_TOP_RX_CTRL_0, reg_gpio21_func_sel, 0); cGpioIndex = 16; #endif #if HV_BOARD_CONFIG_LED_B_POL BOOT_W32_FIELD(CPU_SYS_GPIO0_GPIO_OUT0_SET, gpio0_out_set, (1<>> boot_ddr_init over\n"); Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_INFO, ">>> copy_ddr_code_data...\n"); #ifdef CONFIG_CODE_COMPRESSED extern void *_text_lma,*__text_start; extern void *__data_start; #ifdef CONFIG_USE_DMA Hv_Boot_DmaCopyFlashToDDR(1); #else Hv_Boot_CopyDdrCodeDataCompressed(1); #endif iRet = Hv_Boot_LzssDecompress((char *)DDR_TEMP_ADDR, ((char *)&__text_start), 1, 0); if ( iRet != 0) { Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_INFO, ">>> decompress code error\n"); } #else #ifdef CONFIG_USE_DMA Hv_Boot_DmaCopyFlashToDDR(0); #else Hv_Boot_CopyDdrCodeDataCompressed(0); #endif #endif #ifdef CONFIG_PQ_COMPRESSED #ifdef CONFIG_USE_DMA Hv_Boot_CopyPqFlashToDDR(); Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_DEBUG, ">>> decompress pq...\n"); Hv_Boot_LzssDecompress((char *)DDR_TEMP_ADDR,((char *)HV_MEMORY_CONFIG_PQ_DATA_START), 1, 0); #endif #endif Hv_Boot_ClearBss(); Hv_Boot_CopyLogoFlashToDDR(); /* set stack to the symbol of __stack */ __asm__ __volatile__ ("la $29, __stack"); Hv_Boot_InitTlb(); Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_INFO, ">>> risc init\n"); Hv_Vos_Init(); Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_WAR, ">>> enter main"); iRet = main(0, 0); Hv_Boot_Print_String(BOOT_DEBUG_LEVEL_INFO, ">>> over\n"); return iRet; }