/*---------------------------------------------------------------------------* Project: TwlFirm - nandfirm - menu-launcher File: main.c Copyright 2007 Nintendo. All rights reserved. These coded instructions, statements, and computer programs contain proprietary information of Nintendo of America Inc. and/or Nintendo Company Ltd., and are protected by Federal copyright law. They may not be disclosed to third parties or copied or duplicated in any form, in whole or in part, without the prior written consent of Nintendo. $Date:: $ $Rev$ $Author$ *---------------------------------------------------------------------------*/ #include #include /* PROFILE_ENABLE を定義するとある程度のパフォーマンスチェックができます。 利用するためには、main.cかどこかに、u32 profile[256]; u32 pf_cnt = 0; を 定義する必要があります。 */ //#define PROFILE_ENABLE /* デバッグLEDをFINALROMとは別にOn/Offできます。 */ //#define USE_DEBUG_LED /* PRINT_MEMORY_ADDR を定義すると、そのアドレスからSPrintfを行います(このファイルのみ) FINALROM版でもコードが残るので注意してください。 */ //#define PRINT_MEMORY_ADDR 0x02FFC8A0 /* AES鍵設定API */ extern void SYSMi_SetAESKeysForAccessControlCore( ROM_Header *pROMH, u8 *pDst, BOOL *pIsClearSlotB, BOOL *pIsClearSlotC ); #ifdef PROFILE_ENABLE #define PROFILE_MAX 16 u32 profile[PROFILE_MAX]; u32 pf_cnt = 0; #define PUSH_PROFILE() (profile[pf_cnt++] = (u32)OS_TicksToMicroSeconds(OS_GetTick())) #else #define PUSH_PROFILE() ((void)0) #endif #ifdef USE_DEBUG_LED static u8 step = 0x80; #define InitDebugLED() I2Ci_WriteRegister(I2C_SLAVE_DEBUG_LED, 0x03, 0x00) #define SetDebugLED(pattern) I2Ci_WriteRegister(I2C_SLAVE_DEBUG_LED, 0x01, (pattern)); #else #define InitDebugLED() ((void)0) #define SetDebugLED(pattern) ((void)0) #endif #ifdef PRINT_MEMORY_ADDR static char* debugPtr = (char*)PRINT_MEMORY_ADDR; #undef OS_TPrintf //#define OS_TPrintf(...) (debugPtr = (char*)((u32)(debugPtr + STD_TSPrintf(debugPtr, __VA_ARGS__) + 0xf) & ~0xf)) #define OS_TPrintf(...) (debugPtr += STD_TSPrintf(debugPtr, __VA_ARGS__)) #endif #define THREAD_PRIO_AES 12 #define THREAD_PRIO_FATFS 8 #define DMA_FATFS_1 0 #define DMA_FATFS_2 1 static ROM_Header* const rh= (ROM_Header*)HW_TWL_ROM_HEADER_BUF; /*************************************************************** PreInit FromBootの対応&メインメモリの初期化 OS_Init前なので注意 (ARM9によるメインメモリ初期化で消されないように注意) ***************************************************************/ static void PreInit(void) { /* FromBrom関連 */ if ( !OSi_FromBromToMenu() ) { OS_Terminate(); } /* リセットパラメータ(1バイト)を共有領域(1バイト)にコピー */ #define HOTSTART_FLAG_ENABLE 0x80 *(u8 *)HW_NAND_FIRM_HOTSTART_FLAG = (u8)(MCUi_ReadRegister( (u16)(MCU_REG_TEMP_ADDR + OS_MCU_RESET_VALUE_OFS) ) | HOTSTART_FLAG_ENABLE); } /*************************************************************** PostInit 各種初期化 ***************************************************************/ static void PostInit(void) { /* AES関連 (NAND暗号化の鍵変更を含む) */ if ( OSi_GetFromFirmAddr()->aes_key[2][0] ) { AESi_PreInitKeys(); } // AESの初期化 AES_Init(THREAD_PRIO_AES); // for encrypted NAND // 4: after AES_Init PUSH_PROFILE(); SetDebugLED(++step); // 0x85 // マウント情報の初期化 FS_InitMountInfo(TRUE, FALSE); // 5: after CreateIdleThread PUSH_PROFILE(); SetDebugLED(++step); // 0x86 /* バッテリー残量チェック */ MCUi_WriteRegister( MCU_REG_MODE_ADDR, MCU_SYSTEMMODE_TWL ); // TWL mode for ES library // 6: after MCUi_WriteRegister PUSH_PROFILE(); SetDebugLED(++step); // 0x87 if ( (MCUi_ReadRegister( MCU_REG_POWER_INFO_ADDR ) & MCU_REG_POWER_INFO_LEVEL_MASK) == 0 ) { #ifndef SDK_FINALROM OS_TPanic("Battery is empty.\n"); #else PM_Shutdown(); #endif } } /*************************************************************** EraseAll 不正終了しました いろいろ消してください DSモードにして終わるのがよいか? ***************************************************************/ static void EraseAll(void) { AESi_ResetAesKeyA(); AESi_ResetAesKeyB(); AESi_ResetAesKeyC(); MI_CpuClearFast( OSi_GetFromFirmAddr(), sizeof(OSFromFirmBuf) ); #ifdef SDK_FINALROM MI_CpuClearFast( (void*)HW_TWL_ROM_HEADER_BUF, HW_TWL_ROM_HEADER_BUF_SIZE ); OS_BootFromFIRM(); #endif } void TwlSpMain( void ) { int fd; // menu file descriptor #ifdef PROFILE_ENABLE // 0: bootrom profile[pf_cnt++] = OS_TicksToMicroSecondsBROM32(OS_GetTick()); #endif InitDebugLED(); SetDebugLED(++step); // 0x81 PreInit(); #ifdef PROFILE_ENABLE // 1: after PreInit profile[pf_cnt++] = OS_TicksToMicroSecondsBROM32(OS_GetTick()); #endif SetDebugLED(++step); // 0x82 OS_InitFIRM(); // 2: after OS_InitFIRM PUSH_PROFILE(); SetDebugLED(++step); // 0x83 OS_EnableIrq(); OS_EnableInterrupts(); // 3: after OS_EnableIrq PUSH_PROFILE(); SetDebugLED(++step); // 0x84 PostInit(); // 7: after PostInit PUSH_PROFILE(); #ifdef PROFILE_ENABLE step = 0x88; #endif SetDebugLED(step); // 0x88 // PM_BackLightOn( FALSE ); if ( !FATFS_Init( DMA_FATFS_1, DMA_FATFS_2, THREAD_PRIO_FATFS ) ) { OS_TPrintf("Failed to call FATFS_Init().\n"); goto end; } // 8: after FATFS_Init PUSH_PROFILE(); SetDebugLED(++step); // 0x89 // PM_BackLightOn( FALSE ); if ( PXI_RecvID() != FIRM_PXI_ID_SET_PATH ) { OS_TPrintf("PXI_RecvID() was received invalid value (!=FIRM_PXI_ID_SET_PATH).\n"); goto end; } // 9: after PXI PUSH_PROFILE(); SetDebugLED(++step); // 0x8a // PM_BackLightOn( FALSE ); if ( (fd = FS_OpenSrl()) < 0 ) { OS_TPrintf("Failed to call FS_OpenSrl().\n"); goto end; } // 10: after FS_OpenSrl PUSH_PROFILE(); SetDebugLED(++step); // 0x8b // PM_BackLightOn( FALSE ); if ( !FS_LoadHeader( fd ) ) { OS_TPrintf("Failed to call FS_LoadHeader().\n"); goto end; } // 11: after FS_LoadHeader PUSH_PROFILE(); SetDebugLED(++step); // 0x8c // PM_BackLightOn( FALSE ); if ( PXI_RecvID() != FIRM_PXI_ID_DONE_HEADER ) { OS_TPrintf("PXI_RecvID() was received invalid value (!=FIRM_PXI_ID_DONE_HEADER).\n"); goto end; } // 12: after PXI PUSH_PROFILE(); SetDebugLED(++step); // 0x8d // PM_BackLightOn( FALSE ); AESi_InitKeysFIRM(); AESi_InitSeed(); // 13: after AESi_InitSeed PUSH_PROFILE(); SetDebugLED(++step); // 0x8e // PM_BackLightOn( FALSE ); if ( !FS_LoadStatic( fd ) ) { OS_TPrintf("Failed to call FS_LoadStatic().\n"); goto end; } // 14: after FS_LoadStatic PUSH_PROFILE(); SetDebugLED(++step); // 0x8f // PM_BackLightOn( FALSE ); if ( PXI_RecvID() != FIRM_PXI_ID_DONE_STATIC ) { OS_TPrintf("PXI_RecvID() was received invalid value (!=FIRM_PXI_ID_DONE_STATIC).\n"); goto end; } // 15: after PXI PUSH_PROFILE(); #ifdef PROFILE_ENABLE { int i; PXI_RecvID(); OS_TPrintf("\n[ARM7] Begin\n"); for (i = 0; i < PROFILE_MAX; i++) { // OS_TPrintf("0x%08X\n", profile[i]); if ( !profile[i] ) break; OS_TPrintf("%2d: %7d usec", i, profile[i]); if (i) { OS_TPrintf(" ( %7d usec )\n", profile[i]-profile[i-1]); } else { OS_TPrintf("\n"); } } OS_TPrintf("\n[ARM7] End\n"); } #endif SetDebugLED( 0 ); // PM_BackLightOn( TRUE ); // last chance SYSMi_SetAESKeysForAccessControlCore( (ROM_Header *)HW_TWL_ROM_HEADER_BUF, (u8 *)HW_LAUNCHER_DELIVER_PARAM_BUF, NULL, NULL ); OS_BootFromFIRM(); end: SetDebugLED( (u8)(0xF0 | step)); PM_BackLightOn( TRUE ); EraseAll(); // failed PXI_NotifyID( FIRM_PXI_ID_ERR ); PXI_NotifyID( FIRM_PXI_ID_ERR ); PXI_NotifyID( FIRM_PXI_ID_ERR ); PXI_NotifyID( FIRM_PXI_ID_ERR ); OS_Terminate(); }