/*---------------------------------------------------------------------------* Project: TwlSDK - OS File: os_china.c Copyright 2005-2009 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:: 2009-08-20#$ $Rev: 10998 $ $Author: yada $ *---------------------------------------------------------------------------*/ #include #include #include #include #include #include static char checkString[] = OS_BURY_STRING_FORCHINA; #include #if defined(SDK_ARM9) && defined(SDK_TWLLTD) #include #endif /*****************************************************************************/ /* Constants */ /* Display fade and stop frame count */ #define OSi_FORCHINA_FADE_SPAN 32 #define OSi_FORCHINA_DISP_SPAN 128 /* Index to data images managed by using table */ typedef enum { IMAGE_NOTES_CHR, IMAGE_NOTES_SCR, IMAGE_NOTES_DIGIT, IMAGE_LOGO_CHR, IMAGE_LOGO_SCR, IMAGE_LOGO_PAL, IMAGE_MAX } ImageIndex; /* ONLY_FORCHINA setting */ #define OSi_ONLY_FORCHINA_CHAR_WIDTH 15 #define OSi_ONLY_FORCHINA_CHAR_HEIGHT 2 #define OSi_ONLY_FORCHINA_DOT_WIDTH 117 #define OSi_ONLY_FORCHINA_DOT_HEIGHT 16 #define OSi_ONLY_FORCHINA_CHAR_SIZE 240 /* Character data for startup limitation screen (0-fixed pack of 3 bits into 4 bits) */ static u8 only_forChina_charData[OSi_ONLY_FORCHINA_CHAR_SIZE] ATTRIBUTE_ALIGN(4) = { 0x00, 0x7c, 0xfe, 0xc6, 0xc6, 0xc6, 0xc6, 0xc6, 0x00, 0x8c, 0x8c, 0x9c, 0x9c, 0xbc, 0xbc, 0xec, 0x00, 0x19, 0x19, 0x19, 0x19, 0x19, 0x19, 0x19, 0x00, 0xc3, 0xc3, 0x66, 0x66, 0x3c, 0x3c, 0x18, 0x00, 0xc0, 0xc0, 0xc0, 0xc0, 0xc0, 0xc0, 0xc0, 0x00, 0xcf, 0xef, 0x60, 0x60, 0x60, 0x67, 0x67, 0x00, 0xc7, 0xcf, 0xcc, 0xcc, 0xcc, 0xcc, 0xcc, 0x00, 0x0f, 0x1f, 0x18, 0x18, 0x18, 0x1f, 0x0f, 0x00, 0x60, 0x60, 0x00, 0x00, 0x60, 0x60, 0x60, 0x00, 0x7c, 0xfe, 0xc6, 0xc6, 0xc6, 0xc6, 0xc6, 0x00, 0x00, 0x00, 0x8c, 0x8c, 0x8c, 0x8c, 0x8c, 0x00, 0x00, 0x00, 0xf1, 0xf9, 0x19, 0x19, 0xf9, 0x00, 0xc0, 0xc0, 0xc1, 0xc3, 0xc3, 0xc3, 0xc3, 0x00, 0x87, 0xcf, 0xdc, 0xd8, 0xd8, 0xd8, 0x98, 0x00, 0x0f, 0x0f, 0x00, 0x00, 0x00, 0x07, 0x0f, 0xc6, 0xc6, 0xc6, 0xfe, 0x7c, 0x00, 0x00, 0x00, 0xec, 0xcc, 0xcc, 0x8c, 0x8c, 0x00, 0x00, 0x00, 0x19, 0x19, 0x19, 0xf9, 0xf9, 0x00, 0x00, 0x00, 0x18, 0x18, 0x18, 0x19, 0x19, 0x00, 0x00, 0x00, 0xc0, 0xc0, 0xc0, 0xc0, 0xc0, 0x00, 0x00, 0x00, 0x60, 0x60, 0x60, 0xe0, 0xc0, 0x00, 0x00, 0x00, 0xcc, 0xcc, 0xcc, 0xcf, 0xc7, 0x00, 0x00, 0x00, 0x06, 0x0c, 0x0c, 0x18, 0x18, 0x00, 0x00, 0x00, 0x60, 0x60, 0x60, 0x60, 0x60, 0x00, 0x00, 0x00, 0xc6, 0xc6, 0xc6, 0xfe, 0x7c, 0xe0, 0xc0, 0x00, 0x8c, 0x8c, 0x8c, 0xfc, 0xf8, 0x01, 0x01, 0x00, 0xf9, 0x19, 0x19, 0xf9, 0xf1, 0x00, 0x00, 0x00, 0xc3, 0xc0, 0xc0, 0xc3, 0xc3, 0x00, 0x00, 0x00, 0x18, 0x18, 0x1c, 0xcf, 0xc7, 0x00, 0x00, 0x00, 0x0c, 0x0c, 0x0c, 0x0f, 0x07, 0x00, 0x00, 0x00}; /*****************************************************************************/ /* Declaration */ extern void OSi_InitCommon(void); // os_init.c static u8* LoadImage(ImageIndex index, u32 *p_size); static void WaitForNextFrame(void); static void VBlankIntr(void); static void SetISBNString(const char **isbn, BOOL disp_isbn); static void CheckLanguageCode(void); static void CheckDetectFold(void); static void DispExclusiveMessage(void); static void DispLogoAndNotes(const char **isbn); /*****************************************************************************/ /* Functions */ /*---------------------------------------------------------------------------* Name: LoadImage Description: Allocates memory of sufficient size and loads the specified file. Arguments: index: Index to the file to load p_size: Pointer to u32 that stores the file size If not needed, specify NULL, and it will be ignored. Returns: Memory that is allocated inside the call and where the file is loaded. When the OS_ShowAttentionChina function ends, each heap is destroyed with the OS_ClearAlloc function, so you don't need to explicitly deallocate the memory returned here. *---------------------------------------------------------------------------*/ static u8 *LoadImage(ImageIndex index, u32 *p_size) { void *ptr = NULL; #if defined(SDK_CHINA_DATA_FROM_FILE) /* *INDENT-OFF* */ static const char *(path_table[IMAGE_MAX]) = { "notes_forChina_chrData.bin", "notes_forChina_scrData.bin", "digit_chrData.bin", "logo_forChina_chrData.bin", "logo_forChina_scrData.bin", "logo_forChina_palData.bin", }; /* *INDENT-ON* */ const char *path = path_table[index]; FSFile file[1]; FS_InitFile(file); if (!FS_OpenFileEx(file, path, FS_FILEMODE_R)) { OS_TPanic("failed to open file \"%s\"", path); } else { u32 len = FS_GetFileLength(file); ptr = OS_Alloc(len); if (!ptr) { OS_TPanic("failed to allocate memory for image \"%s\"", path); } else { if (p_size) { *p_size = len; } if (FS_ReadFile(file, ptr, (int)len) != len) { OS_TPanic("failed to read file \"%s\"", path); } } (void)FS_CloseFile(file); } #else extern u8 _binary_notes_forChina_chrData_bin[]; extern u8 _binary_notes_forChina_chrData_bin_end[]; extern u8 _binary_notes_forChina_scrData_bin[]; extern u8 _binary_notes_forChina_scrData_bin_end[]; extern u8 _binary_digit_chrData_bin[]; extern u8 _binary_digit_chrData_bin_end[]; extern u8 _binary_logo_forChina_chrData_bin[]; extern u8 _binary_logo_forChina_chrData_bin_end[]; extern u8 _binary_logo_forChina_scrData_bin[]; extern u8 _binary_logo_forChina_scrData_bin_end[]; extern u8 _binary_logo_forChina_palData_bin[]; extern u8 _binary_logo_forChina_palData_bin_end[]; /* *INDENT-OFF* */ static u8 *(ptr_table[]) = { _binary_notes_forChina_chrData_bin, _binary_notes_forChina_scrData_bin, _binary_digit_chrData_bin, _binary_logo_forChina_chrData_bin, _binary_logo_forChina_scrData_bin, _binary_logo_forChina_palData_bin, }; static u8 *(ptr_end_table[]) = { _binary_notes_forChina_chrData_bin_end, _binary_notes_forChina_scrData_bin_end, _binary_digit_chrData_bin_end, _binary_logo_forChina_chrData_bin_end, _binary_logo_forChina_scrData_bin_end, _binary_logo_forChina_palData_bin_end, }; /* *INDENT-ON* */ ptr = ptr_table[index]; if (p_size) { *p_size = (u32)(ptr_end_table[index] - ptr_table[index]); } #endif return (u8 *)ptr; } /*---------------------------------------------------------------------------* Name: WaitForNextFrame Description: Processing when system is opened and closed, V-Blank wait, and sound update processing. Arguments: None. Returns: None. *---------------------------------------------------------------------------*/ static void WaitForNextFrame(void) { /* Confirm that cover is closed */ if (PAD_DetectFold()) { PM_GoSleepMode(PM_TRIGGER_COVER_OPEN | PM_TRIGGER_CARD, 0, 0); } SVC_WaitVBlankIntr(); } /*---------------------------------------------------------------------------* Name: VBlankIntr Description: V-Blank interrupt vector in exclusive startup control screen. Arguments: None. Returns: None. *---------------------------------------------------------------------------*/ static void VBlankIntr(void) { OS_SetIrqCheckFlag(OS_IE_V_BLANK); } /*---------------------------------------------------------------------------* Name: SetISBNString Description: Applies the ISBN number (and other information) to a prescribed location in the screen data. Arguments: isbn: String array relating to the ISBN number (and other information) disp_isbn: If this is TRUE, display the ISBN number; if it is FALSE, hide the ISBN number. Returns: None. Note: A single character is equivalent to two 8x16 characters. The actual character sizes are as follows. '-': 5x10 '0'-'9': 7x10 *---------------------------------------------------------------------------*/ static void SetISBNString(const char **isbn, BOOL disp_isbn) { s32 i, j; const struct { u8 x, y; /* Leading display offset */ u8 length; /* Display character count */ } pos[] = { { 80, 142, 17} , /* ISBN */ { 125, 157, 12} , /* Joint registration number */ { 106, 173, 4} , /* New sound output tube (left) */ { 139, 173, 3} , /* New sound output tube (right) */ } ; const int count = sizeof(pos) / sizeof(*pos); /* Total number of lines */ const int digit_num = 12; /* Number of characters for the digit display ("0123456789- ") */ u16 pos_x; u16 pos_y; u16 index; u16 width; GXOamAttr *dst = (GXOamAttr *)HW_DB_OAM; for (i = disp_isbn ? 0 : 1; i < count; i++) { pos_x = pos[i].x; pos_y = pos[i].y; for (j = 0; j < pos[i].length; ++j) { if (isbn[i][j] == '-') { index = 10; width = 5; } else if ((isbn[i][j] >= '0') && (isbn[i][j] <= '9')) { index = (u16)(isbn[i][j] - '0'); width = 7; } else { index = 11; width = 7; } dst[0].attr01 = (u32)((pos_x << 16) | (pos_y << 0)); dst[0].attr2 = index; dst[1].attr01 = (u32)((pos_x << 16) | (pos_y + 8 << 0)); dst[1].attr2 = (u16)(index + digit_num); pos_x += width; dst += 2; } } } /*---------------------------------------------------------------------------* Name: OS_InitChina Description: Initializes SDK's OS library. Exclusively for applications for the China region. Use this function instead of OS_Init function. Arguments: isbn: String array relating to the ISBN number (and other information) { char ISBN[ 17 ] , char Joint registration code [ 12 ] , char New output sound pipe (left)[ 4 ] , char New output sound pipe (right) [ 4 ] , } param: Behavior related to displaying ISBN numbers Returns: None. *---------------------------------------------------------------------------*/ void OS_InitChina(const char **isbn, OSChinaIsbn param) { SDK_REFER_SYMBOL(checkString); #ifdef SDK_ARM9 //---- System shared area check SDK_ASSERT((u32)&(OS_GetSystemWork()->command_area) == HW_CMD_AREA); #endif // SDK_ARM9 //---- Initialize SDK OS common OSi_InitCommon(); //---- Check if possible to boot // CheckLanguageCode(); Eliminated starting restrictions //---- Display logo and notes about health care OS_ShowAttentionChina(isbn, param); } /*---------------------------------------------------------------------------* Name: CheckLanguageCode Description: Confirms whether to perform exclusive startup control based on language code. Arguments: None. Returns: None. *---------------------------------------------------------------------------*/ static void CheckLanguageCode(void) { NVRAMConfig *src; src = (NVRAMConfig *)(OS_GetSystemWork()->nvramUserInfo); // Confirms whether it has been determined that language code should not be booted by ARM7 if ((src->ncd.version == 0xff) && (src->ncd.option.language == 7)) { DispExclusiveMessage(); // Never return } } /*---------------------------------------------------------------------------* Name: DispExclusiveMessage Description: Indicates that exclusive startup control was performed based on language code. Arguments: None. Returns: None. *---------------------------------------------------------------------------*/ static void DispExclusiveMessage(void) { // Stop display reg_GX_DISPCNT = 0; reg_GXS_DB_DISPCNT = 0; // Initialize power control if (!(reg_GX_POWCNT & REG_GX_POWCNT_LCD_MASK)) { // When changing LCD enable from OFF to ON, wait 100 ms SVC_WaitByLoop(HW_CPU_CLOCK_ARM9 / 40); } reg_GX_POWCNT = (u16)(REG_GX_POWCNT_DSEL_MASK | REG_GX_POWCNT_E2DG_MASK | REG_GX_POWCNT_E2DGB_MASK | REG_GX_POWCNT_LCD_MASK); // Initialization of master brightness reg_GX_MASTER_BRIGHT = 0; reg_GXS_DB_MASTER_BRIGHT = 0; // Set main LCD { reg_GX_VRAMCNT_A = (u8)((1 << REG_GX_VRAMCNT_A_MST_SHIFT) | (1 << REG_GX_VRAMCNT_A_E_SHIFT)); reg_G2_BG0CNT = (u16)((GX_BG_SCRSIZE_TEXT_256x256 << REG_G2_BG0CNT_SCREENSIZE_SHIFT) | (GX_BG_COLORMODE_16 << REG_G2_BG0CNT_COLORMODE_SHIFT) | (GX_BG_SCRBASE_0xf000 << REG_G2_BG0CNT_SCREENBASE_SHIFT) | (GX_BG_CHARBASE_0x00000 << REG_G2_BG0CNT_CHARBASE_SHIFT) | (GX_BG_EXTPLTT_01 << REG_G2_BG0CNT_BGPLTTSLOT_SHIFT) | (0 << REG_G2_BG0CNT_PRIORITY_SHIFT)); reg_G2_BG0HOFS = (u16)(-(256 - OSi_ONLY_FORCHINA_DOT_WIDTH) / 2); reg_G2_BG0VOFS = (u16)(-(192 - OSi_ONLY_FORCHINA_DOT_HEIGHT) / 2 + 2); reg_GX_DISPCNT |= ((GX_BGMODE_0 << REG_GX_DISPCNT_BGMODE_SHIFT) | (GX_PLANEMASK_BG0 << REG_GX_DISPCNT_DISPLAY_SHIFT)); // Load character data into VRAM { MIUnpackBitsParam param = { OSi_ONLY_FORCHINA_CHAR_SIZE, 1, 4, 0, 0 }; SVC_UnpackBits(only_forChina_charData, (u32 *)(HW_BG_VRAM + 0x20), ¶m); } // Load screen data into VRAM { s32 i; s32 j; u16 code = 1; u16 *dst = (u16 *)(HW_BG_VRAM + 0xf000); for (i = 0; i < OSi_ONLY_FORCHINA_CHAR_HEIGHT; i++) { for (j = 0; j < OSi_ONLY_FORCHINA_CHAR_WIDTH; j++) { *dst++ = code++; } dst += (0x20 - OSi_ONLY_FORCHINA_CHAR_WIDTH); } } } // Set sub LCD { reg_GX_VRAMCNT_C = (u8)((4 << REG_GX_VRAMCNT_C_MST_SHIFT) | (1 << REG_GX_VRAMCNT_C_E_SHIFT)); reg_G2S_DB_BG0CNT = (u16)((GX_BG_SCRSIZE_TEXT_256x256 << REG_G2S_DB_BG0CNT_SCREENSIZE_SHIFT) | (GX_BG_COLORMODE_16 << REG_G2S_DB_BG0CNT_COLORMODE_SHIFT) | (GX_BG_SCRBASE_0xf000 << REG_G2S_DB_BG0CNT_SCREENBASE_SHIFT) | (GX_BG_CHARBASE_0x00000 << REG_G2S_DB_BG0CNT_CHARBASE_SHIFT) | (GX_BG_EXTPLTT_01 << REG_G2S_DB_BG0CNT_BGPLTTSLOT_SHIFT) | (0 << REG_G2S_DB_BG0CNT_PRIORITY_SHIFT)); reg_G2S_DB_BG0OFS = 0; reg_GXS_DB_DISPCNT |= ((GX_BGMODE_0 << REG_GXS_DB_DISPCNT_BGMODE_SHIFT) | (GX_PLANEMASK_BG0 << REG_GXS_DB_DISPCNT_DISPLAY_SHIFT)); } // V-Blank interrupt settings (void)OS_SetIrqFunction(OS_IE_V_BLANK, VBlankIntr); (void)OS_EnableIrqMask(OS_IE_V_BLANK); (void)OS_EnableIrq(); reg_GX_DISPSTAT |= REG_GX_DISPSTAT_VBI_MASK; // Start display reg_GX_DISPCNT |= (u32)(GX_DISPMODE_GRAPHICS << REG_GX_DISPCNT_MODE_SHIFT); reg_GXS_DB_DISPCNT |= (u32)(REG_GXS_DB_DISPCNT_MODE_MASK); { u16 data = 0x001f; u16 target = 0x0001 << 5; s32 i; // Cycle palette colors between red and yellow and loop indefinitely while (TRUE) { for (i = 0; i < 31; i++) { // Confirm that cover is closed if (TRUE == PAD_DetectFold()) { // Turn off power (void)PM_ForceToPowerOff(); } OS_WaitVBlankIntr(); *((u16 *)(HW_BG_PLTT + 2)) = data; data += target; } target = (u16)(~target + 1); } } } /*---------------------------------------------------------------------------* Name: OS_ShowAttentionChina Description: Displays logo for China and notice for prescribed time. Arguments: isbn: String array relating to the ISBN number (and other information) param: Behavior related to displaying ISBN numbers Returns: None. *---------------------------------------------------------------------------*/ SDK_WEAK_SYMBOL void OS_ShowAttentionChina(const char **isbn, OSChinaIsbn param) { /* Structure for saving register */ struct { u32 gx_dispcnt; u32 gxs_dispcnt; u16 gx_powcnt; u16 gx_dispstat; u16 gx_bright; u16 gxs_bright; u16 gx_bg0cnt; u16 gxs_bg0cnt; u32 gx_bg0ofs; u32 gxs_bg0ofs; u8 gx_vramcnt_a; u8 gx_vramcnt_c; u8 gx_vramcnt_d; u8 reserved[1]; OSIrqFunction irqFunction; OSIrqMask irqMask; BOOL irq; OSIntrMode interrupts; } shelter; /* Pre-processing */ { /* Save each register */ shelter.gx_dispstat = reg_GX_DISPSTAT; shelter.gx_dispcnt = reg_GX_DISPCNT; shelter.gxs_dispcnt = reg_GXS_DB_DISPCNT; shelter.gx_bright = reg_GX_MASTER_BRIGHT; shelter.gxs_bright = reg_GXS_DB_MASTER_BRIGHT; shelter.gx_powcnt = reg_GX_POWCNT; shelter.gx_vramcnt_a = reg_GX_VRAMCNT_A; shelter.gx_vramcnt_c = reg_GX_VRAMCNT_C; shelter.gx_vramcnt_d = reg_GX_VRAMCNT_D; shelter.gx_bg0cnt = reg_G2_BG0CNT; shelter.gxs_bg0cnt = reg_G2S_DB_BG0CNT; shelter.gx_bg0ofs = reg_G2_BG0OFS; shelter.gxs_bg0ofs = reg_G2S_DB_BG0OFS; /* Stop display */ reg_GX_DISPCNT = 0; reg_GXS_DB_DISPCNT = 0; /* Initialize power control */ if (!(shelter.gx_powcnt & REG_GX_POWCNT_LCD_MASK)) { /* When changing LCD enable from OFF to ON, wait 100 ms */ SVC_WaitByLoop(HW_CPU_CLOCK_ARM9 / 40); } reg_GX_POWCNT = (u16)(REG_GX_POWCNT_DSEL_MASK | REG_GX_POWCNT_E2DG_MASK | REG_GX_POWCNT_E2DGB_MASK | REG_GX_POWCNT_LCD_MASK); /* Initialization of master brightness */ reg_GX_MASTER_BRIGHT = (u16)((1 << REG_GX_MASTER_BRIGHT_E_MOD_SHIFT) | 16); reg_GXS_DB_MASTER_BRIGHT = reg_GX_MASTER_BRIGHT; /* Prepare display of logo screen */ { reg_GX_VRAMCNT_A = (u8)((1 << REG_GX_VRAMCNT_A_MST_SHIFT) | (1 << REG_GX_VRAMCNT_A_E_SHIFT)); reg_G2_BG0CNT = (u16)((GX_BG_SCRSIZE_TEXT_256x256 << REG_G2_BG0CNT_SCREENSIZE_SHIFT) | (GX_BG_COLORMODE_256 << REG_G2_BG0CNT_COLORMODE_SHIFT) | (GX_BG_SCRBASE_0xf000 << REG_G2_BG0CNT_SCREENBASE_SHIFT) | (GX_BG_CHARBASE_0x00000 << REG_G2_BG0CNT_CHARBASE_SHIFT) | (GX_BG_EXTPLTT_01 << REG_G2_BG0CNT_BGPLTTSLOT_SHIFT) | (0 << REG_G2_BG0CNT_PRIORITY_SHIFT)); reg_G2_BG0OFS = 0; reg_GX_DISPCNT |= ((GX_BGMODE_0 << REG_GX_DISPCNT_BGMODE_SHIFT) | (GX_PLANEMASK_BG0 << REG_GX_DISPCNT_DISPLAY_SHIFT)); /* Load character data into VRAM */ { void *const src = LoadImage(IMAGE_LOGO_CHR, NULL); MI_UncompressLZ16(src, (u32 *)HW_BG_VRAM); } /* Load screen data into VRAM */ { u16 *src = (u16*)LoadImage(IMAGE_LOGO_SCR, NULL); u16 *dst; s32 i; s32 j; // Place 12x12 (96x96-pixel) screen data at the center SVC_CpuClearFast(0, (u32 *)(HW_BG_VRAM + 0xf000), 32*24*2); dst = (u16*)(HW_BG_VRAM + 0xf000 + ((6*32+10)*2)); for (i = 0; i < 12; i++, dst += 32) { for (j = 0; j < 12; j++, src++) { dst[j] = *src; } } } /* Load palette data into VRAM */ { u32 size; void *const src = LoadImage(IMAGE_LOGO_PAL, &size); SVC_CpuCopyFast(src, (u32 *)(HW_BG_PLTT), size); } } /* Prepare to display Notes screen */ { reg_GX_VRAMCNT_C = (u8)((4 << REG_GX_VRAMCNT_C_MST_SHIFT) | (1 << REG_GX_VRAMCNT_C_E_SHIFT)); reg_GX_VRAMCNT_D = (u8)((4 << REG_GX_VRAMCNT_D_MST_SHIFT) | (1 << REG_GX_VRAMCNT_D_E_SHIFT)); reg_G2S_DB_BG0CNT = (u16)((GX_BG_SCRSIZE_TEXT_256x256 << REG_G2S_DB_BG0CNT_SCREENSIZE_SHIFT) | (GX_BG_COLORMODE_16 << REG_G2S_DB_BG0CNT_COLORMODE_SHIFT) | (GX_BG_SCRBASE_0xf000 << REG_G2S_DB_BG0CNT_SCREENBASE_SHIFT) | (GX_BG_CHARBASE_0x00000 << REG_G2S_DB_BG0CNT_CHARBASE_SHIFT) | (GX_BG_EXTPLTT_01 << REG_G2S_DB_BG0CNT_BGPLTTSLOT_SHIFT) | (0 << REG_G2S_DB_BG0CNT_PRIORITY_SHIFT)); reg_G2S_DB_BG0OFS = 0; reg_GXS_DB_DISPCNT |= ((GX_BGMODE_0 << REG_GXS_DB_DISPCNT_BGMODE_SHIFT) | ((GX_PLANEMASK_BG0 | GX_PLANEMASK_OBJ) << REG_GXS_DB_DISPCNT_DISPLAY_SHIFT)); /* Load character data into VRAM */ { u32 size; void *src; /* Background text */ src = LoadImage(IMAGE_NOTES_CHR, &size); MI_UncompressLZ16(src, (u32 *)(HW_DB_BG_VRAM)); /* OBJ text */ src = LoadImage(IMAGE_NOTES_DIGIT, &size); MI_UncompressLZ16(src, (u32 *)(HW_DB_OBJ_VRAM)); SVC_CpuClear(0xC0, (void *)HW_DB_OAM, HW_OAM_SIZE, 32); } /* Load screen data into VRAM */ { u32 size; void *const src = LoadImage(IMAGE_NOTES_SCR, &size); SVC_CpuCopyFast(src, (u32 *)(HW_DB_BG_VRAM + 0xf000), size); } // Display settings for the ISBN string SetISBNString(isbn, (BOOL)(param != OS_CHINA_ISBN_NO_DISP)); /* Edit palette */ *((u16 *)(HW_DB_BG_PLTT + 0)) = 0x7fff; *((u16 *)(HW_DB_BG_PLTT + 2)) = 0x0000; *((u16 *)(HW_DB_OBJ_PLTT + 0)) = 0x7fff; *((u16 *)(HW_DB_OBJ_PLTT + 2)) = 0x0000; } /* Start display */ reg_GX_DISPCNT |= (u32)(GX_DISPMODE_GRAPHICS << REG_GX_DISPCNT_MODE_SHIFT); reg_GXS_DB_DISPCNT |= (u32)(REG_GXS_DB_DISPCNT_MODE_MASK); /* Interrupt settings */ reg_GX_DISPSTAT |= REG_GX_DISPSTAT_VBI_MASK; shelter.irqFunction = OS_GetIrqFunction(OS_IE_V_BLANK); OS_SetIrqFunction(OS_IE_V_BLANK, VBlankIntr); shelter.irqMask = OS_EnableIrqMask(OS_IE_V_BLANK); shelter.irq = OS_EnableIrq(); shelter.interrupts = OS_EnableInterrupts(); } /* Main process */ { s32 i; u16 pad_old; u16 pad; /* Fade in */ for (i = 0; i < OSi_FORCHINA_FADE_SPAN; i++) { reg_GX_MASTER_BRIGHT = (u16)((1 << REG_GX_MASTER_BRIGHT_E_MOD_SHIFT) | 16 - ((i + 1) * 16 / OSi_FORCHINA_FADE_SPAN)); reg_GXS_DB_MASTER_BRIGHT = reg_GX_MASTER_BRIGHT; WaitForNextFrame(); } /* Skip based on button input */ pad_old = PAD_Read(); for (i = 0; i < OSi_FORCHINA_DISP_SPAN; i++) { pad = PAD_Read(); if ((param == OS_CHINA_ISBN_CHECK_ROM) && (~pad_old & pad)) { /* If any button push trigger was detected, skip */ break; } pad_old = pad; WaitForNextFrame(); // On ROMs used for censorship purposes, do not exit until a button is pressed if(param == OS_CHINA_ISBN_CHECK_ROM) i = 0; } /* Fade out */ for (i = 0; i < OSi_FORCHINA_FADE_SPAN; i++) { reg_GX_MASTER_BRIGHT = (u16)((1 << REG_GX_MASTER_BRIGHT_E_MOD_SHIFT) | (i + 1) * 16 / OSi_FORCHINA_FADE_SPAN); reg_GXS_DB_MASTER_BRIGHT = reg_GX_MASTER_BRIGHT; WaitForNextFrame(); } } /* Post-processing */ { /* Stop display */ reg_GX_DISPCNT &= ~(GX_DISPMODE_GRAPHICS << REG_GX_DISPCNT_MODE_SHIFT); reg_GXS_DB_DISPCNT &= ~REG_GXS_DB_DISPCNT_MODE_MASK; /* Restore interrupt-related items */ (void)OS_RestoreInterrupts(shelter.interrupts); (void)OS_RestoreIrq(shelter.irq); (void)OS_SetIrqMask(shelter.irqMask); OS_SetIrqFunction(OS_IE_V_BLANK, shelter.irqFunction); reg_GX_DISPSTAT &= ~REG_GX_DISPSTAT_VBI_MASK; /* Clear used VRAM (cannot restore initial values because the size is too big) */ SVC_CpuClearFast(0, (u32 *)HW_BG_VRAM, 0x10000); SVC_CpuClearFast(0, (u32 *)HW_BG_PLTT, 0x200); SVC_CpuClearFast(0, (u32 *)HW_DB_BG_VRAM, 0x10000); SVC_CpuClearFast(0, (u32 *)HW_DB_BG_PLTT, 0x04); SVC_CpuClearFast(0, (u32 *)HW_DB_OAM, HW_OAM_SIZE); /* Restore each register */ reg_G2S_DB_BG0OFS = shelter.gxs_bg0ofs; reg_G2_BG0OFS = shelter.gx_bg0ofs; reg_G2S_DB_BG0CNT = shelter.gxs_bg0cnt; reg_G2_BG0CNT = shelter.gx_bg0cnt; reg_GX_VRAMCNT_D = shelter.gx_vramcnt_d; reg_GX_VRAMCNT_C = shelter.gx_vramcnt_c; reg_GX_VRAMCNT_A = shelter.gx_vramcnt_a; reg_GX_POWCNT = shelter.gx_powcnt; if (!(shelter.gx_powcnt & REG_GX_POWCNT_LCD_MASK)) { /* When changing LCD enable from ON to OFF, wait 100 ms */ SVC_WaitByLoop(HW_CPU_CLOCK_ARM9 / 40); } reg_GXS_DB_MASTER_BRIGHT = shelter.gxs_bright; reg_GX_MASTER_BRIGHT = shelter.gx_bright; reg_GXS_DB_DISPCNT = shelter.gxs_dispcnt; reg_GX_DISPCNT = shelter.gx_dispcnt; reg_GX_DISPSTAT = shelter.gx_dispstat; } } /*---------------------------------------------------------------------------* End of file *---------------------------------------------------------------------------*/